React v19

2024年12月5日 作者:React团队


注意

React 19 现在已稳定发布!

自四月初与React 19 RC版本一同分享此文章以来的新增内容

此文章的日期已更新,以反映稳定版本的发布日期。

React v19 现在已在 npm 上可用!

在我们的React 19 升级指南中,我们分享了将您的应用程序升级到 React 19 的分步说明。在这篇文章中,我们将概述 React 19 的新功能以及如何采用它们。

有关重大更改的列表,请参阅升级指南


React 19 的新增功能

操作

React 应用中一个常见的用例是执行数据变异,然后相应地更新状态。例如,当用户提交表单更改其姓名时,您将发出 API 请求,然后处理响应。过去,您需要手动处理挂起状态、错误、乐观更新和顺序请求。

例如,您可以在useState中处理挂起和错误状态。

// Before Actions
function UpdateName({}) {
const [name, setName] = useState("");
const [error, setError] = useState(null);
const [isPending, setIsPending] = useState(false);

const handleSubmit = async () => {
setIsPending(true);
const error = await updateName(name);
setIsPending(false);
if (error) {
setError(error);
return;
}
redirect("/path");
};

return (
<div>
<input value={name} onChange={(event) => setName(event.target.value)} />
<button onClick={handleSubmit} disabled={isPending}>
Update
</button>
{error && <p>{error}</p>}
</div>
);
}

在 React 19 中,我们增加了对在转换中使用异步函数的支持,以自动处理挂起状态、错误、表单和乐观更新。

例如,您可以使用useTransition为您处理挂起状态。

// Using pending state from Actions
function UpdateName({}) {
const [name, setName] = useState("");
const [error, setError] = useState(null);
const [isPending, startTransition] = useTransition();

const handleSubmit = () => {
startTransition(async () => {
const error = await updateName(name);
if (error) {
setError(error);
return;
}
redirect("/path");
})
};

return (
<div>
<input value={name} onChange={(event) => setName(event.target.value)} />
<button onClick={handleSubmit} disabled={isPending}>
Update
</button>
{error && <p>{error}</p>}
</div>
);
}

异步转换将立即将isPending状态设置为 true,发出异步请求,并在任何转换后将isPending切换为 false。这允许您在数据更改时保持当前 UI 的响应性和交互性。

注意

按照惯例,使用异步转换的函数称为“操作”。

操作会自动为您管理数据的提交。

  • 挂起状态:操作提供一个挂起状态,该状态在请求开始时启动,并在提交最终状态更新时自动重置。
  • 乐观更新:操作支持新的useOptimistic hook,因此您可以在请求提交时向用户显示即时反馈。
  • 错误处理:操作提供错误处理,以便您在请求失败时显示错误边界,并自动将乐观更新还原到其原始值。
  • 表单<form>元素现在支持将函数传递给actionformAction属性。将函数传递给action属性默认使用操作,并在提交后自动重置表单。

基于操作,React 19 引入了useOptimistic 来管理乐观更新,以及一个新的 hook React.useActionState 来处理操作的常见情况。在react-dom中,我们添加了<form>操作来自动管理表单,并添加了useFormStatus来支持表单中操作的常见情况。

在 React 19 中,上面的示例可以简化为

// Using <form> Actions and useActionState
function ChangeName({ name, setName }) {
const [error, submitAction, isPending] = useActionState(
async (previousState, formData) => {
const error = await updateName(formData.get("name"));
if (error) {
return error;
}
redirect("/path");
return null;
},
null,
);

return (
<form action={submitAction}>
<input type="text" name="name" />
<button type="submit" disabled={isPending}>Update</button>
{error && <p>{error}</p>}
</form>
);
}

在下一节中,我们将分解 React 19 中每个新的操作功能。

新的 hook:useActionState

为了使操作的常见情况更容易,我们添加了一个名为useActionState的新 hook。

const [error, submitAction, isPending] = useActionState(
async (previousState, newName) => {
const error = await updateName(newName);
if (error) {
// You can return any result of the action.
// Here, we return only the error.
return error;
}

// handle success
return null;
},
null,
);

useActionState 接收一个函数(“Action”),并返回一个包装后的 Action 用于调用。这是因为 Actions 可以组合。当调用包装后的 Action 时,useActionState 将返回 Action 的最后结果作为 data,以及 Action 的挂起状态作为 pending

注意

React.useActionState 之前在 Canary 版本中被称为 ReactDOM.useFormState,但我们已对其进行了重命名,并弃用了 useFormState

更多信息请参见 #28491

更多信息,请参见 useActionState 的文档。

React DOM: <form> Actions

Actions 也与 React 19 的新 <form> 功能集成到 react-dom 中。我们添加了对将函数作为 actionformAction 属性传递到 <form><input><button> 元素的支持,以便使用 Actions 自动提交表单。

<form action={actionFunction}>

<form> Action 成功时,React 将自动重置非受控组件的表单。如果您需要手动重置 <form>,您可以调用新的 requestFormReset React DOM API。

更多信息,请参见 react-dom 的文档,了解 <form><input><button>

React DOM: 新的 Hook: useFormStatus

在设计系统中,通常会编写需要访问其所在 <form> 信息的设计组件,而无需将 props 传递到组件。这可以通过 Context 完成,但为了使常见情况更容易,我们添加了一个新的 Hook useFormStatus

import {useFormStatus} from 'react-dom';

function DesignButton() {
const {pending} = useFormStatus();
return <button type="submit" disabled={pending} />
}

useFormStatus 读取父 <form> 的状态,就像表单是一个 Context 提供者一样。

更多信息,请参见 react-dom 的文档,了解 useFormStatus

新的 Hook: useOptimistic

执行数据变异时,另一种常见的 UI 模式是在异步请求进行时乐观地显示最终状态。在 React 19 中,我们添加了一个名为 useOptimistic 的新 Hook 来简化此操作。

function ChangeName({currentName, onUpdateName}) {
const [optimisticName, setOptimisticName] = useOptimistic(currentName);

const submitAction = async formData => {
const newName = formData.get("name");
setOptimisticName(newName);
const updatedName = await updateName(newName);
onUpdateName(updatedName);
};

return (
<form action={submitAction}>
<p>Your name is: {optimisticName}</p>
<p>
<label>Change Name:</label>
<input
type="text"
name="name"
disabled={currentName !== optimisticName}
/>
</p>
</form>
);
}

useOptimistic Hook 将立即渲染 optimisticName,同时 updateName 请求正在进行中。当更新完成或出错时,React 将自动切换回 currentName 值。

更多信息,请参见 useOptimistic 的文档。

新的 API: use

在 React 19 中,我们引入了一个新的 API 来在渲染中读取资源:use

例如,您可以使用 use 读取 Promise,React 将暂停直到 Promise 解析。

import {use} from 'react';

function Comments({commentsPromise}) {
// `use` will suspend until the promise resolves.
const comments = use(commentsPromise);
return comments.map(comment => <p key={comment.id}>{comment}</p>);
}

function Page({commentsPromise}) {
// When `use` suspends in Comments,
// this Suspense boundary will be shown.
return (
<Suspense fallback={<div>Loading...</div>}>
<Comments commentsPromise={commentsPromise} />
</Suspense>
)
}

注意

use 不支持在渲染过程中创建的 Promise。

如果你尝试将渲染过程中创建的 Promise 传递给use,React 将发出警告。

控制台
一个组件被未缓存的 Promise 挂起。目前不支持在客户端组件或钩子中创建 Promise,除非通过兼容 Suspense 的库或框架。

要修复此问题,你需要从支持 Promise 缓存的 Suspense 驱动库或框架传递 Promise。未来我们计划推出一些特性,以便更容易在渲染过程中缓存 Promise。

你也可以使用use读取上下文,允许你在特定条件下(例如早期返回之后)读取上下文。

import {use} from 'react';
import ThemeContext from './ThemeContext'

function Heading({children}) {
if (children == null) {
return null;
}

// This would not work with useContext
// because of the early return.
const theme = use(ThemeContext);
return (
<h1 style={{color: theme.color}}>
{children}
</h1>
);
}

use API 只能在渲染过程中调用,类似于钩子。与钩子不同,use 可以有条件地调用。未来我们计划支持更多使用use在渲染过程中使用资源的方法。

更多信息,请参阅 use 的文档。

新的 React DOM 静态 API

我们为静态站点生成添加了两个新的 API 到react-dom/static

这些新的 API 在renderToString的基础上进行了改进,它等待数据加载以生成静态 HTML。它们旨在与 Node.js 流和 Web 流等流式环境一起使用。例如,在 Web 流环境中,你可以使用prerender将 React 树预渲染为静态 HTML。

import { prerender } from 'react-dom/static';

async function handler(request) {
const {prelude} = await prerender(<App />, {
bootstrapScripts: ['/main.js']
});
return new Response(prelude, {
headers: { 'content-type': 'text/html' },
});
}

预渲染 API 将等待所有数据加载完毕后再返回静态 HTML 流。流可以转换为字符串,也可以发送流式响应。它们不支持在内容加载时进行流式传输,而现有的React DOM 服务器渲染 API 支持。

更多信息,请参阅React DOM 静态 API

React 服务端组件

服务端组件

服务端组件是一种新的选项,允许在打包之前,在与你的客户端应用程序或 SSR 服务器不同的环境中提前渲染组件。这个单独的环境就是 React 服务端组件中的“服务器”。服务端组件可以在你的 CI 服务器上构建时运行一次,或者可以使用 Web 服务器为每个请求运行。

React 19 包含 Canary 频道中包含的所有 React 服务端组件功能。这意味着带有服务端组件的库现在可以将 React 19 作为对等依赖项,并使用react-server 导出条件 用于支持完整堆栈 React 架构 的框架中。

注意

如何构建对服务端组件的支持?

虽然 React 19 中的 React 服务端组件是稳定的,并且在主要版本之间不会中断,但用于实现 React 服务端组件打包器或框架的基础 API 不遵循语义版本控制,并且在 React 19.x 的次要版本之间可能会中断。

为了支持 React 服务端组件作为打包器或框架,我们建议固定到特定的 React 版本,或使用 Canary 版本。我们将继续与打包器和框架合作,以稳定将来用于实现 React 服务端组件的 API。

更多信息,请参阅 React 服务端组件 的文档。

服务端操作

服务端操作允许客户端组件调用在服务器上执行的异步函数。

当使用"use server"指令定义服务端操作时,你的框架将自动创建一个对服务器函数的引用,并将该引用传递给客户端组件。当在客户端调用该函数时,React 将向服务器发送请求以执行该函数并返回结果。

注意

服务端组件没有指令。

一个常见的误解是服务端组件由"use server"表示,但是服务端组件没有指令。"use server"指令用于服务端操作。

更多信息,请参阅 指令 的文档。

服务端操作可以在服务端组件中创建并作为 props 传递给客户端组件,或者可以在客户端组件中导入和使用。

更多信息,请参阅 React 服务端操作 的文档。

React 19 的改进

ref 作为属性

从 React 19 开始,您现在可以访问函数组件的ref 属性。

function MyInput({placeholder, ref}) {
return <input placeholder={placeholder} ref={ref} />
}

//...
<MyInput ref={ref} />

新的函数组件将不再需要forwardRef,我们将发布一个 codemod 来自动更新您的组件以使用新的ref 属性。在未来的版本中,我们将弃用并移除forwardRef

注意

传递给类的refs 不会作为属性传递,因为它们引用组件实例。

水合错误的差异

我们还改进了 react-dom 中水合错误的错误报告。例如,在 DEV 中,不再记录多个错误而没有任何关于不匹配的信息

控制台
警告:文本内容不匹配。服务器:“服务器” 客户端:“客户端” at span at App
警告:水合过程中发生错误。
中的服务器 HTML 已被客户端内容替换。
警告:文本内容不匹配。服务器:“服务器” 客户端:“客户端” at span at App
警告:水合过程中发生错误。
中的服务器 HTML 已被客户端内容替换。
未捕获的错误:文本内容与服务器端渲染的 HTML 不匹配。 at checkForUnmatchedText

我们现在记录一条包含不匹配差异的单一消息

控制台
未捕获的错误:由于服务器端渲染的 HTML 与客户端不匹配,导致水合失败。因此,这棵树将在客户端重新生成。如果使用了服务器端渲染的客户端组件,则可能发生这种情况 - 服务器/客户端分支 if (typeof window !== 'undefined')。 - 每次调用都会更改的变量输入,例如 Date.now()Math.random()。 - 与服务器不匹配的用户区域设置中的日期格式。 - 未随 HTML 一起发送其快照的外部更改数据。 - 无效的 HTML 标签嵌套。 如果客户端安装了与 React 加载前弄乱 HTML 的浏览器扩展,也可能发生这种情况。 https://reactjs.ac.cn/link/hydration-mismatch <App> <span>+ 客户端- 服务器 at throwOnHydrationMismatch

<Context> 作为提供程序

在 React 19 中,您可以将 <Context> 渲染为提供程序,而不是 <Context.Provider>

const ThemeContext = createContext('');

function App({children}) {
return (
<ThemeContext value="dark">
{children}
</ThemeContext>
);
}

新的 Context 提供程序可以使用 <Context>,我们将发布一个 codemod 来转换现有的提供程序。在未来的版本中,我们将弃用 <Context.Provider>

refs 的清理函数

我们现在支持从ref 回调中返回清理函数。

<input
ref={(ref) => {
// ref created

// NEW: return a cleanup function to reset
// the ref when element is removed from DOM.
return () => {
// ref cleanup
};
}}
/>

当组件卸载时,React 将调用从ref 回调返回的清理函数。这适用于 DOM refs、类组件的 refs 和useImperativeHandle

注意

以前,React 在卸载组件时会使用null 调用ref 函数。如果您的ref 返回一个清理函数,React 现在将跳过此步骤。

在未来的版本中,我们将弃用在卸载组件时使用null 调用 refs。

由于引入了 ref 清理函数,现在 TypeScript 将拒绝从ref 回调返回任何其他内容。解决方法通常是停止使用隐式返回,例如

- <div ref={current => (instance = current)} />
+ <div ref={current => {instance = current}} />

原始代码返回了HTMLDivElement 的实例,而 TypeScript 无法知道这是否是 *应该* 是清理函数,或者您是否不想返回清理函数。

您可以使用 no-implicit-ref-callback-return codemod 此模式。

useDeferredValue 初始值

我们在useDeferredValue 中添加了一个initialValue 选项。

function Search({deferredValue}) {
// On initial render the value is ''.
// Then a re-render is scheduled with the deferredValue.
const value = useDeferredValue(deferredValue, '');

return (
<Results query={value} />
);
}

当提供 initialValue 时,useDeferredValue 将在组件的初始渲染中将其作为value 返回,并在后台调度重新渲染,并返回 deferredValue

更多信息,请参见 useDeferredValue

文档元数据支持

在 HTML 中,文档元数据标签,例如 <title><link><meta>,都必须放置在文档的 <head> 部分。在 React 中,决定哪些元数据适合应用的组件可能离你渲染 <head> 的位置很远,或者 React 根本不渲染 <head>。过去,这些元素需要在 effect 中手动插入,或者使用 react-helmet 等库来插入,并且在服务器端渲染 React 应用时需要小心处理。

在 React 19 中,我们增加了对在组件中原生渲染文档元数据标签的支持。

function BlogPost({post}) {
return (
<article>
<h1>{post.title}</h1>
<title>{post.title}</title>
<meta name="author" content="Josh" />
<link rel="author" href="https://twitter.com/joshcstory/" />
<meta name="keywords" content={post.keywords} />
<p>
Eee equals em-see-squared...
</p>
</article>
);
}

当 React 渲染此组件时,它将看到 <title><link><meta> 标签,并自动将它们提升到文档的 <head> 部分。通过原生支持这些元数据标签,我们可以确保它们与仅客户端应用、流式 SSR 和服务器组件一起工作。

注意

你可能仍然需要一个元数据库

对于简单的用例,将文档元数据渲染为标签可能很合适,但库可以提供更强大的功能,例如根据当前路由使用特定元数据覆盖通用元数据。这些功能使 react-helmet 等框架和库更容易支持元数据标签,而不是替换它们。

更多信息,请参阅 <title><link><meta> 的文档。

样式表支持

样式表,无论是外部链接的(<link rel="stylesheet" href="...">)还是内联的(<style>...</style>),由于样式优先级规则,需要在 DOM 中小心放置。构建允许在组件内组合使用的样式表功能很困难,因此用户最终往往要么远离可能依赖它们的组件加载所有样式,要么使用封装此复杂性的样式库。

在 React 19 中,我们解决了这个复杂性问题,并通过内置的样式表支持,提供了对客户端并发渲染和服务器端流式渲染的更深入集成。如果你告诉 React 你的样式表的 precedence(优先级),它将管理样式表在 DOM 中的插入顺序,并确保在显示依赖这些样式规则的内容之前加载样式表(如果为外部样式表)。

function ComponentOne() {
return (
<Suspense fallback="loading...">
<link rel="stylesheet" href="foo" precedence="default" />
<link rel="stylesheet" href="bar" precedence="high" />
<article class="foo-class bar-class">
{...}
</article>
</Suspense>
)
}

function ComponentTwo() {
return (
<div>
<p>{...}</p>
<link rel="stylesheet" href="baz" precedence="default" /> <-- will be inserted between foo & bar
</div>
)
}

在服务器端渲染期间,React 将样式表包含在 <head> 中,确保浏览器在加载样式表之前不会进行绘制。如果在开始流式传输后才发现样式表,React 将确保在显示依赖该样式表的 Suspense 边界的内容之前,将样式表插入到客户端的 <head> 中。

在客户端渲染期间,React 将等待新渲染的样式表加载完毕后再提交渲染。如果你在应用程序中多个位置渲染此组件,React 只会在文档中包含样式表一次。

function App() {
return <>
<ComponentOne />
...
<ComponentOne /> // won't lead to a duplicate stylesheet link in the DOM
</>
}

对于习惯于手动加载样式表的用户来说,这是一个机会,可以将这些样式表与依赖它们的组件放在一起,从而实现更好的局部推理,并更容易确保只加载实际依赖的样式表。

样式库和与捆绑器的样式集成也可以采用此新功能,因此即使你没有直接渲染自己的样式表,你仍然可以在你的工具升级到使用此功能时从中受益。

更多详情,请阅读 <link><style> 的文档。

异步脚本支持

在 HTML 中,普通脚本(<script src="...">)和延迟脚本(<script defer="" src="...">)按照文档顺序加载,这使得在组件树深处渲染这些类型的脚本具有挑战性。但是,异步脚本(<script async="" src="...">)将以任意顺序加载。

在 React 19 中,我们改进了对异步脚本的支持,允许您在组件树中的任何位置渲染它们,在实际依赖脚本的组件内部,而无需管理脚本实例的重新定位和去重。

function MyComponent() {
return (
<div>
<script async={true} src="..." />
Hello World
</div>
)
}

function App() {
<html>
<body>
<MyComponent>
...
<MyComponent> // won't lead to duplicate script in the DOM
</body>
</html>
}

在所有渲染环境中,异步脚本都将被去重,以便 React 只加载并执行一次脚本,即使它是被多个不同的组件渲染的。

在服务器端渲染中,异步脚本将包含在`<head>`中,并且优先级低于更重要的、会阻塞绘制的资源,例如样式表、字体和图片预加载。

更多详情,请阅读`<script>` 的文档。

预加载资源支持

在初始文档加载和客户端更新期间,尽早告知浏览器可能需要加载的资源,可以极大地影响页面性能。

React 19 包含许多用于加载和预加载浏览器资源的新 API,使构建不会因资源加载效率低下而受阻的出色体验变得尽可能容易。

import { prefetchDNS, preconnect, preload, preinit } from 'react-dom'
function MyComponent() {
preinit('https://.../path/to/some/script.js', {as: 'script' }) // loads and executes this script eagerly
preload('https://.../path/to/font.woff', { as: 'font' }) // preloads this font
preload('https://.../path/to/stylesheet.css', { as: 'style' }) // preloads this stylesheet
prefetchDNS('https://...') // when you may not actually request anything from this host
preconnect('https://...') // when you will request something but aren't sure what
}
<!-- the above would result in the following DOM/HTML -->
<html>
<head>
<!-- links/scripts are prioritized by their utility to early loading, not call order -->
<link rel="prefetch-dns" href="https://...">
<link rel="preconnect" href="https://...">
<link rel="preload" as="font" href="https://.../path/to/font.woff">
<link rel="preload" as="style" href="https://.../path/to/stylesheet.css">
<script async="" src="https://.../path/to/some/script.js"></script>
</head>
<body>
...
</body>
</html>

这些 API 可用于优化初始页面加载,方法是将附加资源(如字体)的发现从样式表加载中移出。它们还可以通过预取预期导航使用的资源列表,然后在点击甚至悬停时急切地预加载这些资源,从而加快客户端更新速度。

更多详情,请参阅 资源预加载 API

与第三方脚本和扩展的兼容性

我们改进了水合过程,以适应第三方脚本和浏览器扩展。

在水合过程中,如果在客户端渲染的元素与从服务器HTML中找到的元素不匹配,React将强制进行客户端重新渲染以修正内容。以前,如果元素是由第三方脚本或浏览器扩展插入的,它会触发不匹配错误和客户端渲染。

在 React 19 中,`<head>` 和 `<body>` 中的意外标签将被跳过,避免不匹配错误。如果 React 需要由于不相关的 hydration 错配而重新渲染整个文档,它将保留第三方脚本和浏览器扩展插入的样式表。

更好的错误报告

我们改进了 React 19 中的错误处理,以消除重复并提供处理已捕获和未捕获错误的选项。例如,如果错误边界捕获了渲染中的错误,以前 React 会抛出两次错误(一次针对原始错误,另一次在未能自动恢复后),然后使用有关错误发生位置的信息调用`console.error`。

这导致每个捕获的错误都会产生三个错误。

控制台
未捕获的错误:hit at Throws at renderWithHooks
未捕获的错误:hit <-- 重复 at Throws at renderWithHooks
上述错误发生在 Throws 组件中 at Throws at ErrorBoundary at App React 将尝试使用您提供的错误边界 ErrorBoundary 从头开始重新创建此组件树。

在 React 19 中,我们记录单个错误,其中包含所有错误信息。

控制台
错误:hit at Throws at renderWithHooks 上述错误发生在 Throws 组件中 at Throws at ErrorBoundary at App React 将尝试使用您提供的错误边界 ErrorBoundary 从头开始重新创建此组件树。 at ErrorBoundary at App

此外,我们添加了两个新的根选项来补充`onRecoverableError`。

  • onCaughtError:当 React 在错误边界中捕获错误时调用。
  • onUncaughtError:当抛出错误且未被错误边界捕获时调用。
  • onRecoverableError:当抛出错误并自动恢复时调用。

有关更多信息和示例,请参阅`createRoot` 和 `hydrateRoot` 的文档。

自定义元素支持

React 19 完全支持自定义元素,并在 Custom Elements Everywhere 上通过了所有测试。

在过去的版本中,在 React 中使用自定义元素很困难,因为 React 将无法识别的 props 视为属性而不是特性。在 React 19 中,我们添加了对特性的支持,该特性在客户端和 SSR 期间均有效,策略如下:

  • 服务器端渲染:如果传递给自定义元素的 props 类型是原始值,例如`string`、`number` 或值为 `true`,则将渲染为属性。具有非原始类型的 props,例如`object`、`symbol`、`function` 或值为 `false` 将被省略。
  • 客户端渲染:与自定义元素实例上的特性匹配的 props 将被分配为特性,否则将被分配为属性。

感谢 Joey Arhar 推动 React 中自定义元素支持的设计和实现。

如何升级

请参阅 React 19 升级指南,了解分步说明以及完整的重大更改和值得注意的更改列表。

注意:本文最初发表于 2024年4月25日,并已于 2024年12月5日更新为稳定版本。