'use client'
'use client'
标记在客户端运行的代码。
参考
'use client'
在文件顶部添加 'use client'
以将模块及其传递的依赖项标记为客户端代码。
'use client';
import { useState } from 'react';
import { formatDate } from './formatters';
import Button from './button';
export default function RichTextEditor({ timestamp, text }) {
const date = formatDate(timestamp);
// ...
const editButton = <Button />;
// ...
}
当从服务器组件导入带有 'use client'
标记的文件时,兼容的捆绑工具 将模块导入视为服务器运行和客户端运行代码之间的边界。
作为 RichTextEditor
的依赖项,无论 formatDate
与 Button
的模块是否包含 'use client'
,其都将在客户端上进行评估。请注意,当从服务器代码导入时,单个模块可能在服务器上进行评估,并且当从客户端代码导入时,可能在客户端上进行评估。
注意
'use client'
必须位于文件的最前方,位于任何导入或其他代码之上(可以位于代码顶部的注释之下)。它们必须用单引号或双引号编写,但不能用反引号。- 当从另一个客户端渲染的模块导入
'use client'
模块时,该指令不起作用。 - 当组件模块包含
'use client'
指令时,保证对该组件的任何使用都将是客户端组件。然而,即使没有'use client'
指令,组件仍可以在客户端上进行评估。- 如果在带有
'use client'
指令的模块中定义了组件使用,那么该使用将被视为客户端组件,或者当它是包含'use client'
指令的模块的传递依赖时。否则,它将是服务器组件。
- 如果在带有
- 标记为客户端评估的代码不仅限于组件。客户端模块子树的所有代码都将被发送到客户端并由客户端运行。
- 当服务器评估的模块从
'use client'
模块导入值时,这些值必须是 React 组件或 受支持的可序列化属性值,以传递给客户端组件。其他用例将引发异常。
'use client'
如何标记客户端代码
在 React 应用程序中,组件通常被拆分为单独的文件,或称之为 模块。
对于使用 React 服务器组件的应用程序,默认情况下是由服务器渲染的。'use client'
引入了 模块依赖树 中的服务器客户端边界,有效地创建了一个客户端模块的子树。
为了更好地说明这一点,请参考下面的 React 服务器组件应用程序示例。
import FancyText from './FancyText'; import InspirationGenerator from './InspirationGenerator'; import Copyright from './Copyright'; export default function App() { return ( <> <FancyText title text="Get Inspired App" /> <InspirationGenerator> <Copyright year={2004} /> </InspirationGenerator> </> ); }
在这个示例应用程序的模块依赖树中,InspirationGenerator.js
中的 'use client'
指令标记了该模块及其所有传递依赖为客户端模块。从 InspirationGenerator.js
开始的子树现在被标记为客户端模块。
在渲染过程中,框架将服务器渲染根组件,并继续通过 渲染树 进行渲染,选择不评估从客户端标记的代码导入的任何代码。
然后,渲染树的服务器渲染部分将发送到客户端。客户端,具有其客户端代码下载,然后完成渲染树的其余部分。
我们引入以下定义:
- 客户端组件 是渲染树中在客户端上渲染的组件。
- 服务器组件 是渲染树中在服务器上渲染的组件。
通过示例应用程序,App
、FancyText
和 Copyright
都是服务器渲染的,被视为服务器组件。由于 InspirationGenerator.js
及其传递依赖被标记为客户端代码,组件 InspirationGenerator
及其子组件 FancyText
都是客户端组件。
深入探讨
根据上述定义,组件 FancyText
如何实现既是服务器组件又是客户端组件的呢?
首先,术语“组件”并不是非常精确的。以下是“组件”可以被理解的两种方式:
- “组件”可以指的是 组件定义。在大多数情况下,这将是一个函数。
// 这是组件的定义
function MyComponent() {
return <p>My Component</p>
}
- “组件”可以指的是定义的 组件的使用。
import MyComponent from './MyComponent';
function App() {
// 这是组件的用法
return <MyComponent />;
}
通常,在解释概念时,不太精确的性质并不重要,但在这种情况下是重要的。
当谈论服务器组件或客户端组件时,我们指的是组件的使用。
- 如果组件在带有
'use client'
指令的模块中定义,或者组件在客户端组件中导入并调用,那么组件的使用将是客户端组件。 - 否则,组件的使用将是服务器组件。
回到关于 FancyText
的问题,我们可以看到组件定义没有包含 'use client'
指令,并且它有两个使用方式。
将 FancyText
作为 App
的子组件使用,将该使用标记为服务器组件。当 FancyText
在 InspirationGenerator
下导入并调用时,FancyText
的这种使用是客户端组件,因为 InspirationGenerator
包含 'use client'
指令。
这意味着 FancyText
的组件定义将在服务器上进行评估,同时也将被客户端下载以渲染其客户端组件的使用。
深入探讨
由于 Copyright
是作为客户端组件 InspirationGenerator
的子组件进行渲染,这可能会让人感到惊讶,它却是一个服务器组件。
请回想一下,'use client'
定义了在 模块依赖树 上的服务器和客户端代码的边界,而不是在渲染树上。
在模块依赖树中可以看到 App.js
从 Copyright.js
模块导入并调用 Copyright
。由于 Copyright.js
没有包含 'use client'
指令,该组件的使用在服务器上进行渲染。App
作为根组件在服务器上进行渲染。
由于可以将 JSX 作为 props 传递,客户端组件可以渲染服务器组件。在这种情况下,InspirationGenerator
以 children 的形式接收 Copyright
。然而,InspirationGenerator
模块从未直接导入 Copyright
模块,也不调用该组件,所有这些都是由 App
完成的。实际上,在 InspirationGenerator
开始渲染之前,Copyright
组件已经完全执行完毕。
请牢记,组件之间的父子渲染关系并不保证相同的渲染环境。
何时使用 'use client'
?
通过 'use client'
可以确定哪些组件是客户端组件。由于服务器组件是默认的,以下是关于服务器组件的优势和限制的简要概述,以确定何时需要将某些内容标记为客户端渲染。
为简单起见,我们只谈论了服务器组件,但相同的原则适用于应用程序中的所有在服务器上运行的代码。
服务器组件的优点
- 服务器组件可以减少客户端发送和运行的代码量。只有客户端模块会被捆绑和由客户端进行评估。
- 服务器组件受益于在服务器上运行。它们可以访问本地文件系统,并且可能在数据获取和网络请求方面体验较低的延迟。
服务器组件的限制
- 服务器组件不能支持交互,因为事件处理程序必须由客户端注册和触发。
- 例如,像
onClick
这样的事件处理程序只能在客户端组件中定义。
- 例如,像
- 服务器组件不能使用大多数 Hooks。
- 当服务器组件被渲染时,它们的输出基本上是客户端渲染的一系列组件。服务器组件在渲染后不会在内存中保留,并且不能拥有自己的状态。
服务器组件返回的可序列化类型
与任何 React 应用程序一样,父组件会向子组件传递数据。由于它们在不同的环境中渲染,因此需要额外考虑将数据从服务器组件传递到客户端组件。
从服务器组件传递给客户端组件的属性值必须是可序列化的。
可序列化属性包括:
- 原始类型
- 包含可序列化值的迭代类型
- Date
- 普通 对象:使用 对象初始化器 创建的、具有可序列化属性
- Server Action 中的函数
- 客户端或服务器组件元素(JSX)
- Promise
值得注意的是,以下内容不受支持:
- 未从客户端标记的模块中导出或未标记为
'use server'
的 函数 - 类
- 任何类的实例对象(除了提到的内置类)或具有 null 原型 的对象
- 未全局注册的符号,例如
Symbol('my new symbol')
用法
构建交互性与状态
'use client'; import { useState } from 'react'; export default function Counter({initialValue = 0}) { const [countValue, setCountValue] = useState(initialValue); const increment = () => setCountValue(countValue + 1); const decrement = () => setCountValue(countValue - 1); return ( <> <h2>Count Value: {countValue}</h2> <button onClick={increment}>+1</button> <button onClick={decrement}>-1</button> </> ); }
由于 Counter
需要使用 useState
Hook 和事件处理程序来增加或减少值,因此该组件必须是客户端组件,并且需要在顶部添加 'use client'
指令。
相比之下,一个渲染 UI 而没有交互的组件不需要成为客户端组件。
import { readFile } from 'node:fs/promises';
import Counter from './Counter';
export default async function CounterContainer() {
const initialValue = await readFile('/path/to/counter_value');
return <Counter initialValue={initialValue} />
}
例如,Counter
的父组件 CounterContainer
不需要 'use client'
,因为它没有交互并且不使用状态。此外,CounterContainer
必须是服务器组件,因为它在服务器上从本地文件系统读取数据,这仅在服务器组件中可行。
还有一些组件不使用任何特定于服务器或客户端的功能,可以在渲染位置上保持中立。在之前的示例中,FancyText
就是这样一个组件。
export default function FancyText({title, text}) {
return title
? <h1 className='fancy title'>{text}</h1>
: <h3 className='fancy cursive'>{text}</h3>
}
在这种情况下,不添加 'use client'
指令的结果是,当从服务器组件引用时,FancyText
的输出(而不是其源代码)将被发送到浏览器。正如在之前的 Inspirations 应用程序示例中演示的,FancyText
既可以作为服务器组件也可以作为客户端组件使用,具体取决于它被导入和使用的位置。
但如果 FancyText
的 HTML 输出相对于其源代码(包括依赖项)较大,那么强制将其始终作为客户端组件可能更高效。返回较长 SVG 路径字符串的组件就是可能更高效地强制组件成为客户端组件的一种情况。
使用客户端 API
React 应用程序可能使用特定于客户端的 API,例如浏览器的 web 存储 API、音频和视频处理 API 以及有关设备硬件 API 等 其他 API。
在这个示例中,该组件使用 DOM API 来操作 canvas
元素。由于这些 API 仅在浏览器中可用,因此必须将其标记为客户端组件。
'use client';
import {useRef, useEffect} from 'react';
export default function Circle() {
const ref = useRef(null);
useLayoutEffect(() => {
const canvas = ref.current;
const context = canvas.getContext('2d');
context.reset();
context.beginPath();
context.arc(100, 75, 50, 0, 2 * Math.PI);
context.stroke();
});
return <canvas ref={ref} />;
}
使用第三方库
在 React 应用程序中,通常会利用第三方库来处理常见的 UI 模式或逻辑。
这些库可能依赖于组件 Hook 或客户端 API。使用以下 React API 中的任何一个的第三方组件必须在客户端上运行:
- createContext
react
和react-dom
Hook,但不包括use
和useId
- forwardRef
- memo
- startTransition
- 如果这些库使用了客户端 API,例如向 DOM 插入元素或查看本机平台视图
如果这些库已经更新为与 React 服务器组件兼容,那么它们将已经包含自己的 'use client'
标记,从而使开发者可以直接在服务器组件中使用它们。如果某个库尚未更新,或者某个组件需要像事件处理程序这样只能在客户端上指定的 props,那么可能需要在第三方客户端组件和希望使用它的服务器组件之间添加自己的客户端组件文件。