Next.js 中关于'use client'的使用误解及解法

client component vs server component

我们在 page.tsx中导入 Button 自定义组件

tsx 复制代码
// app/page.tsx
import styles from './page.module.css'
import Button from './components/button'
export default function Home() {
	return (
		<main className={styles.main}>
			<h1>hello world</h1>
			<Button />
		</main>
	)
}

由于 nextjs 默认 src之下的组件都是 server component,在Button组件(server component)中,如果直接调用client component 才能使用的交互性例如 onClick useState是会报错的

可以简单按照报错提示修改为 client component 这样就没问题

但是如果你稍不注意直接在 page.tsx 加上'use client'也是没有报错的!

但是这样造成的问题是:导入page.tsx 的所有组件都将是'client component',那和直接写 react 有什么区别..

tsx 复制代码
'use client'
import styles from './page.module.css'
import Button from './components/button'
import Post from './components/post'

export default function Home() {
	return (
		<main className={styles.main}>
			<h1>hello world</h1>
			<Button />
            <Post/>
		</main>
	)
}

假设 Post 组件中需要加载一个很大的第三方库例如sanitize-html

如果放在client component 就不太适宜,可以看官方文档的第四条,将导入的较大第三方库尽量放在server component上减少客户端的压力

所以最优的解决办法就是:谁需要交互性就将最叶子节点变成客户端组件

✔️

使用 context 封装 children

假设你需要使用一个上下文 context 传递一些数据,你可以这样写

tsx 复制代码
// app/context/ThemeComtext.tsx
'use client'
export default function ThemeContextProvider({ children }: { children: React.ReactNode }) {
	return <div>{children}</div>
}

使用时在 layout 上封装一层 wrapper

tsx 复制代码
import type { Metadata } from 'next'
import { Inter } from 'next/font/google'
import './globals.css'
import ThemeContextProvider from './context/ThemeContext'

const inter = Inter({ subsets: ['latin'] })
export default function RootLayout({ children }: { children: React.ReactNode }) {
	return (
		<html lang="en">
			<body className={inter.className}>
				<ThemeContextProvider>{children}</ThemeContextProvider>
			</body>
		</html>
	)
}

ThemeContextProvider其实是一个客户端组件,如果把 children 根组件 wrapper 在 这个context 中根据上面的说法岂不是之下的所有组件也都是客户端组件了?

正确思考方式是 : 关注import tree 而不是 render tree

上一个例子中,我们在一个使用'use client'的客户端组件中 import其他组件,那么这些组件都会变成客户端组件,而这里的例子其实只是传递组件children,并不会将 children 中的服务端组件改成客户端组件

总结就是 改变组件和 jsx 渲染的结构无关,而是和import 相关联

多次导入交互性组件

将一个交互性的组件分别导入至 client && server 组件会是什么情况?

交互性组件Button

jsx 复制代码
import style from './button.module.css'
export default function Button() {
	return (
		<div className={style.btn} onClick={() => console.log('click me')}>
			Click me
		</div>
	)
}

server Component

jsx 复制代码
import sanitizeHtml from 'sanitize-html'
import Button from './button'
export default function Post() {
	return (
		<div>
			Post 
			<Button />
		</div>
	)
}

Client component

jsx 复制代码
'use client'
import Button from './button'
export default function Form() {
	return (
		<div>
			form <Button />
		</div>
	)
}

结果是符合直觉的 如果将 Post 组件删掉是不会报错的,也就是说在 client component使用Button会表现为client,而在 server component不会被影响,各自独立

参考

相关推荐
用户269948725937011 分钟前
使用命令获取figma节点树JSON文件
前端
三小河12 分钟前
JavaScript 稀疏数组:成因、坑点与解决方案
前端
HelloReader18 分钟前
创建第一个 Qt Quick 应用从零到窗口弹出(四)
前端
三旬818 分钟前
Day.js 源码深度剖析:极简时间库的设计艺术
javascript
HelloReader20 分钟前
Qt 项目构建入门CMake 完全指南(三)
前端
用户9083246027327 分钟前
Spring AI + RAG + SSE 实现带搜索来源的智能问答完整方案
前端·后端
GISer_Jing32 分钟前
阿里开源纯前端浏览器自动化 PageAgent,[特殊字符] 浏览器自动化变天啦?
前端·人工智能·自动化·aigc·交互
清风徐来QCQ1 小时前
js中的模板字符串
开发语言·前端·javascript
成都渲染101云渲染66661 小时前
Houdini+Blender高效渲染方案(高配算力+全渲染器兼容)
前端·系统架构
SuperEugene1 小时前
Vue3 + Element Plus 表格实战:批量操作、行内编辑、跨页选中逻辑统一|表单与表格规范篇
开发语言·前端·javascript