nextjs-server-client-components
Compare original and translation side by side
🇺🇸
Original
English🇨🇳
Translation
ChineseNext.js Server Components vs Client Components
Next.js Server Components 与 Client Components 对比
Overview
概述
Provide comprehensive guidance for choosing between Server Components and Client Components in Next.js App Router, including cookie/header access, searchParams handling, pathname routing, and React's 'use' API for promise unwrapping.
提供Next.js App Router中Server Components与Client Components的全面选择指南,包括cookie/header访问、searchParams处理、pathname路由以及用于解包Promise的React 'use' API。
TypeScript: NEVER Use any
Type
anyTypeScript: 绝对不要使用any
类型
anyCRITICAL RULE: This codebase has enabled. Using will cause build failures.
@typescript-eslint/no-explicit-anyany❌ WRONG:
typescript
function handleSubmit(e: any) { ... }
const data: any[] = [];✅ CORRECT:
typescript
function handleSubmit(e: React.FormEvent<HTMLFormElement>) { ... }
const data: string[] = [];关键规则: 此代码库已启用规则。使用会导致构建失败。
@typescript-eslint/no-explicit-anyany❌ 错误示例:
typescript
function handleSubmit(e: any) { ... }
const data: any[] = [];✅ 正确示例:
typescript
function handleSubmit(e: React.FormEvent<HTMLFormElement>) { ... }
const data: string[] = [];Common Next.js Type Patterns
常见Next.js类型模式
typescript
// Page props
function Page({ params }: { params: { slug: string } }) { ... }
function Page({ searchParams }: { searchParams: { [key: string]: string | string[] | undefined } }) { ... }
// Form events
const handleSubmit = (e: React.FormEvent<HTMLFormElement>) => { ... }
const handleChange = (e: React.ChangeEvent<HTMLInputElement>) => { ... }
// Server actions
async function myAction(formData: FormData) { ... }typescript
// 页面属性
function Page({ params }: { params: { slug: string } }) { ... }
function Page({ searchParams }: { searchParams: { [key: string]: string | string[] | undefined } }) { ... }
// 表单事件
const handleSubmit = (e: React.FormEvent<HTMLFormElement>) => { ... }
const handleChange = (e: React.ChangeEvent<HTMLInputElement>) => { ... }
// 服务器操作
async function myAction(formData: FormData) { ... }When to Use This Skill
何时使用本指南
Use this skill when:
- Deciding whether to use Server or Client Components
- Accessing cookies, headers, or other server-side data
- Working with searchParams or route parameters
- Needing pathname or routing information
- Unwrapping promises with React 'use' API
- Debugging 'use client' boundary issues
- Optimizing component rendering strategy
在以下场景使用本指南:
- 决定使用Server还是Client Components时
- 访问cookies、headers或其他服务器端数据时
- 处理searchParams或路由参数时
- 需要pathname或路由信息时
- 使用React 'use' API解包Promise时
- 调试'use client'边界问题时
- 优化组件渲染策略时
Core Decision: Server vs Client Components
核心决策:Server vs Client Components
Default: Server Components
默认选择:Server Components
All components in the App Router are Server Components by default. No directive needed.
typescript
// app/components/ProductList.tsx
// This is a Server Component (default)
export default async function ProductList() {
const products = await fetch('https://api.example.com/products');
const data = await products.json();
return (
<ul>
{data.map(product => (
<li key={product.id}>{product.name}</li>
))}
</ul>
);
}When to use Server Components:
- Fetching data from APIs or databases
- Accessing backend resources (environment variables, file system)
- Processing sensitive information (API keys, tokens)
- Reducing client-side JavaScript bundle
- SEO-critical content rendering
- Static or infrequently changing content
Benefits:
- Zero client-side JavaScript by default
- Direct database/API access
- Secure handling of secrets
- Automatic code splitting
- Better initial page load performance
- Reduced bundle size
App Router中的所有组件默认都是Server Components,无需添加任何指令。
typescript
// app/components/ProductList.tsx
// 这是一个Server Component(默认)
export default async function ProductList() {
const products = await fetch('https://api.example.com/products');
const data = await products.json();
return (
<ul>
{data.map(product => (
<li key={product.id}>{product.name}</li>
))}
</ul>
);
}何时使用Server Components:
- 从API或数据库获取数据时
- 访问后端资源(环境变量、文件系统)时
- 处理敏感信息(API密钥、令牌)时
- 减少客户端JavaScript包体积时
- 渲染对SEO至关重要的内容时
- 渲染静态或不常变更的内容时
优势:
- 默认无需客户端JavaScript
- 可直接访问数据库/API
- 安全处理机密信息
- 自动代码分割
- 初始页面加载性能更优
- 包体积更小
Client Components: 'use client'
Client Components:使用'use client'指令
Add directive at the top of a file to make it a Client Component.
'use client'typescript
// app/components/Counter.tsx
'use client';
import { useState } from 'react';
export default function Counter() {
const [count, setCount] = useState(0);
return (
<button onClick={() => setCount(count + 1)}>
Count: {count}
</button>
);
}When to use Client Components:
- Need React hooks (useState, useEffect, useContext, etc.)
- Event handlers (onClick, onChange, onSubmit, etc.)
- Browser-only APIs (window, localStorage, navigator)
- Third-party libraries requiring browser environment
- Interactive UI elements (modals, dropdowns, forms)
- Real-time features (WebSocket, animations)
Requirements for Client Components:
- Must have directive at top of file
'use client' - Cannot use async/await directly in component
- Cannot access server-only APIs (cookies, headers)
- All imported components become Client Components
在文件顶部添加指令,即可将组件设为Client Component。
'use client'typescript
// app/components/Counter.tsx
'use client';
import { useState } from 'react';
export default function Counter() {
const [count, setCount] = useState(0);
return (
<button onClick={() => setCount(count + 1)}>
Count: {count}
</button>
);
}何时使用Client Components:
- 需要使用React钩子(useState、useEffect、useContext等)时
- 需要事件处理函数(onClick、onChange、onSubmit等)时
- 需要浏览器专属API(window、localStorage、navigator)时
- 需要依赖浏览器环境的第三方库时
- 需要交互式UI元素(模态框、下拉菜单、表单)时
- 需要实时功能(WebSocket、动画)时
Client Components的要求:
- 必须在文件顶部添加指令
'use client' - 不能在组件中直接使用async/await
- 不能访问服务器专属API(cookies、headers)
- 所有导入的组件都会变为Client Components
⚠️ CRITICAL: Server Components NEVER Need 'use client'
⚠️ 关键注意:Server Components永远不需要'use client'
Server Components are the DEFAULT. DO NOT add 'use client' unless you specifically need client-side features.
✅ CORRECT - Server Component with Navigation:
typescript
// app/page.tsx - Server Component (NO 'use client' needed!)
import Link from 'next/link';
import { redirect } from 'next/navigation';
export default async function Page() {
// Server components can be async
const data = await fetchData();
if (!data) {
redirect('/login'); // Server-side redirect
}
return (
<div>
<Link href="/dashboard">Go to Dashboard</Link>
<p>{data.content}</p>
</div>
);
}❌ WRONG - Adding 'use client' to Server Component:
typescript
// app/page.tsx
'use client'; // ❌ WRONG! Don't add this to server components!
export default async function Page() { // ❌ Will fail - async client components not allowed
const data = await fetchData();
return <div>{data.content}</div>;
}Server Navigation Methods (NO 'use client' needed):
- component from
<Link>next/link - function from
redirect()next/navigation - Server Actions (see Advanced Routing skill)
Client Navigation Methods (REQUIRES 'use client'):
- hook from
useRouter()next/navigation - hook
usePathname() - hook (also requires Suspense)
useSearchParams()
Server Components是默认选项。除非明确需要客户端特性,否则不要添加'use client'。
✅ 正确示例 - 包含导航的Server Component:
typescript
// app/page.tsx - Server Component(无需'use client'!)
import Link from 'next/link';
import { redirect } from 'next/navigation';
export default async function Page() {
// Server Components可以是异步的
const data = await fetchData();
if (!data) {
redirect('/login'); // 服务器端重定向
}
return (
<div>
<Link href="/dashboard">前往仪表盘</Link>
<p>{data.content}</p>
</div>
);
}❌ 错误示例 - 给Server Component添加'use client':
typescript
// app/page.tsx
'use client'; // ❌ 错误!不要给Server Components添加这个!
export default async function Page() { // ❌ 会失败 - 不允许异步Client Components
const data = await fetchData();
return <div>{data.content}</div>;
}服务器端导航方法(无需'use client'):
- 来自的
next/link组件<Link> - 来自的
next/navigation函数redirect() - 服务器操作(参考高级路由指南)
客户端导航方法(需要'use client'):
- 来自的
next/navigation钩子useRouter() - 钩子
usePathname() - 钩子(同时需要Suspense)
useSearchParams()
Server Component Patterns
Server Component模式
Accessing Cookies
访问Cookies
Use to read cookies in Server Components:
next/headerstypescript
// app/dashboard/page.tsx
import { cookies } from 'next/headers';
export default async function Dashboard() {
const cookieStore = await cookies();
const token = cookieStore.get('session-token');
if (!token) {
redirect('/login');
}
const user = await fetchUser(token.value);
return <div>Welcome, {user.name}</div>;
}Important Notes:
- must be awaited in Next.js 15+
cookies() - Cookies are read-only in Server Components
- To set cookies, use Server Actions (see Advanced Routing skill)
- Cookie access is only available in Server Components
在Server Components中使用读取cookies:
next/headerstypescript
// app/dashboard/page.tsx
import { cookies } from 'next/headers';
export default async function Dashboard() {
const cookieStore = await cookies();
const token = cookieStore.get('session-token');
if (!token) {
redirect('/login');
}
const user = await fetchUser(token.value);
return <div>欢迎,{user.name}</div>;
}重要说明:
- Next.js 15+中必须使用await
cookies() - Server Components中Cookies是只读的
- 如需设置Cookies,请使用服务器操作(参考高级路由指南)
- 仅Server Components可访问Cookies
Accessing Headers
访问Headers
typescript
// app/api/route.ts or any Server Component
import { headers } from 'next/headers';
export default async function Page() {
const headersList = await headers();
const userAgent = headersList.get('user-agent');
const referer = headersList.get('referer');
return <div>User Agent: {userAgent}</div>;
}typescript
// app/api/route.ts 或任意Server Component
import { headers } from 'next/headers';
export default async function Page() {
const headersList = await headers();
const userAgent = headersList.get('user-agent');
const referer = headersList.get('referer');
return <div>用户代理:{userAgent}</div>;
}Using searchParams
使用searchParams
Access URL query parameters directly in Server Components:
typescript
// app/search/page.tsx
export default async function SearchPage({
searchParams,
}: {
searchParams: { q?: string; category?: string };
}) {
const query = searchParams.q || '';
const category = searchParams.category || 'all';
const results = await searchProducts(query, category);
return (
<div>
<h1>Search Results for: {query}</h1>
<p>Category: {category}</p>
<ProductList products={results} />
</div>
);
}Important Notes:
- is only available in
searchParamsfilespage.tsx - In Next.js 15+, must be awaited
searchParams - searchParams is NOT available in
layout.tsx - Use client-side hook if needed in Client Components
useSearchParams()
⚠️ CRITICAL WARNING - Next.js 15+ searchParams:
When extracting parameters in Next.js 15+, you MUST use destructuring to keep the identifier visible in the same line as the parameter extraction. Do NOT use intermediate variables like or - this is an anti-pattern that breaks code readability and testing patterns.
searchParamsparamsresolvedAsync searchParams (Next.js 15+):
typescript
// app/search/page.tsx (Next.js 15+)
export default async function SearchPage({
searchParams,
}: {
searchParams: Promise<{ q?: string }>;
}) {
// BEST PRACTICE: Inline access keeps searchParams and parameter together on one line
const q = (await searchParams).q || '';
return <div>Search: {q}</div>;
}CRITICAL PATTERN REQUIREMENT:
When extracting parameters from , ALWAYS use inline access to keep and the parameter name on the SAME LINE:
searchParamssearchParamstypescript
// ✅ CORRECT: Inline access (REQUIRED PATTERN)
const name = (await searchParams).name || '';
// ✅ ALSO CORRECT: Multiple parameters
const category = (await searchParams).category || 'all';
const sort = (await searchParams).sort || 'asc';
// ❌ WRONG: Using intermediate variable separates searchParams from parameter
const params = await searchParams; // DON'T DO THIS
const name = params.name; // searchParams not visible here
// ❌ WRONG: Destructuring (searchParams and name on same line but missing second 'name')
const { name } = await searchParams; // Not preferredWhy inline access:
- Keeps identifier visible on the same line as parameter extraction
searchParams - Makes the relationship between URL parameter and variable explicit
- Satisfies code review and testing patterns that check for proper searchParams usage
在Server Components中直接访问URL查询参数:
typescript
// app/search/page.tsx
export default async function SearchPage({
searchParams,
}: {
searchParams: { q?: string; category?: string };
}) {
const query = searchParams.q || '';
const category = searchParams.category || 'all';
const results = await searchProducts(query, category);
return (
<div>
<h1>搜索结果:{query}</h1>
<p>分类:{category}</p>
<ProductList products={results} />
</div>
);
}重要说明:
- 仅在
searchParams文件中可用page.tsx - Next.js 15+中必须使用await
searchParams - 在
searchParams中不可用layout.tsx - 如需在Client Components中使用,请使用客户端钩子
useSearchParams()
⚠️ 关键警告 - Next.js 15+的searchParams:
在Next.js 15+中提取参数时,必须使用解构方式,确保标识符与参数提取在同一行可见。不要使用或等中间变量,这是违反代码可读性和测试模式的反模式。
searchParamsparamsresolved异步searchParams(Next.js 15+):
typescript
// app/search/page.tsx (Next.js 15+)
export default async function SearchPage({
searchParams,
}: {
searchParams: Promise<{ q?: string }>;
}) {
// 最佳实践:内联访问使searchParams和参数在同一行可见
const q = (await searchParams).q || '';
return <div>搜索:{q}</div>;
}关键模式要求:
从提取参数时,必须始终使用内联访问,确保标识符与参数名称在同一行:
searchParamssearchParamstypescript
// ✅ 正确:内联访问(要求的模式)
const name = (await searchParams).name || '';
// ✅ 正确:多个参数
const category = (await searchParams).category || 'all';
const sort = (await searchParams).sort || 'asc';
// ❌ 错误:使用中间变量分离searchParams与参数
const params = await searchParams; // 不要这样做
const name = params.name; // 此处看不到searchParams
// ❌ 错误:解构(searchParams和name在同一行但缺少第二个'name')
const { name } = await searchParams; // 不推荐为什么使用内联访问:
- 使标识符与参数提取在同一行可见
searchParams - 明确URL参数与变量之间的关系
- 符合代码审查和测试模式对searchParams正确使用的要求
Using pathname and Route Information
使用pathname和路由信息
In Server Components (page.tsx):
typescript
// app/blog/[slug]/page.tsx
export default async function BlogPost({
params,
}: {
params: { slug: string };
}) {
// params contains route parameters
const post = await getPost(params.slug);
return <article>{post.title}</article>;
}Async params (Next.js 15+):
typescript
// app/blog/[slug]/page.tsx (Next.js 15+)
export default async function BlogPost({
params,
}: {
params: Promise<{ slug: string }>;
}) {
const { slug } = await params;
const post = await getPost(slug);
return <article>{post.title}</article>;
}In Client Components:
Use hooks from :
next/navigationtypescript
// app/components/Breadcrumbs.tsx
'use client';
import { usePathname, useParams, useSearchParams } from 'next/navigation';
export default function Breadcrumbs() {
const pathname = usePathname(); // Current path: /blog/hello-world
const params = useParams(); // Route params: { slug: 'hello-world' }
const searchParams = useSearchParams(); // Query params
return (
<nav>
<span>Current path: {pathname}</span>
<span>Slug: {params.slug}</span>
<span>Search: {searchParams.get('q')}</span>
</nav>
);
}在Server Components(page.tsx)中:
typescript
// app/blog/[slug]/page.tsx
export default async function BlogPost({
params,
}: {
params: { slug: string };
}) {
// params包含路由参数
const post = await getPost(params.slug);
return <article>{post.title}</article>;
}异步params(Next.js 15+):
typescript
// app/blog/[slug]/page.tsx (Next.js 15+)
export default async function BlogPost({
params,
}: {
params: Promise<{ slug: string }>;
}) {
const { slug } = await params;
const post = await getPost(slug);
return <article>{post.title}</article>;
}在Client Components中:
使用中的钩子:
next/navigationtypescript
// app/components/Breadcrumbs.tsx
'use client';
import { usePathname, useParams, useSearchParams } from 'next/navigation';
export default function Breadcrumbs() {
const pathname = usePathname(); // 当前路径:/blog/hello-world
const params = useParams(); // 路由参数:{ slug: 'hello-world' }
const searchParams = useSearchParams(); // 查询参数
return (
<nav>
<span>当前路径:{pathname}</span>
<span>Slug:{params.slug}</span>
<span>搜索:{searchParams.get('q')}</span>
</nav>
);
}⚠️ CRITICAL: useSearchParams ALWAYS Requires Suspense
⚠️ 关键注意:useSearchParams始终需要Suspense
When using hook, you MUST:
useSearchParams()- Add directive at the top of the file
'use client' - Wrap the component in a Suspense boundary
This is a Next.js requirement - failing to do both will cause errors.
✅ CORRECT Pattern:
typescript
// app/page.tsx or any parent component
import { Suspense } from 'react';
import SearchComponent from './SearchComponent';
export default function Page() {
return (
<Suspense fallback={<div>Loading...</div>}>
<SearchComponent />
</Suspense>
);
}
// app/SearchComponent.tsx
'use client';
import { useSearchParams } from 'next/navigation';
export default function SearchComponent() {
const searchParams = useSearchParams();
const query = searchParams.get('q');
return <div>Search query: {query}</div>;
}❌ WRONG - Missing 'use client':
typescript
// This will fail - useSearchParams requires 'use client'
import { useSearchParams } from 'next/navigation';
export default function SearchComponent() {
const searchParams = useSearchParams(); // ERROR!
return <div>{searchParams.get('q')}</div>;
}❌ WRONG - Missing Suspense wrapper:
typescript
// This will cause issues - useSearchParams requires Suspense
export default function Page() {
return <SearchComponent />; // Missing Suspense wrapper!
}使用钩子时,必须:
useSearchParams()- 在文件顶部添加指令
'use client' - 用Suspense边界包裹组件
这是Next.js的强制要求 - 不满足这两点会导致错误。
✅ 正确模式:
typescript
// app/page.tsx 或任何父组件
import { Suspense } from 'react';
import SearchComponent from './SearchComponent';
export default function Page() {
return (
<Suspense fallback={<div>加载中...</div>}>
<SearchComponent />
</Suspense>
);
}
// app/SearchComponent.tsx
'use client';
import { useSearchParams } from 'next/navigation';
export default function SearchComponent() {
const searchParams = useSearchParams();
const query = searchParams.get('q');
return <div>搜索关键词:{query}</div>;
}❌ 错误示例 - 缺少'use client':
typescript
// 会失败 - useSearchParams需要'use client'
import { useSearchParams } from 'next/navigation';
export default function SearchComponent() {
const searchParams = useSearchParams(); // 错误!
return <div>{searchParams.get('q')}</div>;
}❌ 错误示例 - 缺少Suspense包裹:
typescript
// 会出现问题 - useSearchParams需要Suspense
export default function Page() {
return <SearchComponent />; // 缺少Suspense包裹!
}React 'use' API for Promise Unwrapping
用于解包Promise的React 'use' API
The React API allows reading promises and context in both Server and Client Components.
useReact API允许在Server和Client Components中读取Promise和上下文。
useUsing 'use' with Promises
结合Promise使用'use'
typescript
// app/components/UserProfile.tsx
'use client';
import { use } from 'react';
// IMPORTANT: Use specific types, generic types, or 'unknown' - NEVER 'any'
// Option 1: Specific type (best when type is known)
export default function UserProfile({
userPromise
}: {
userPromise: Promise<{ name: string; email: string }>
}) {
// Unwrap the promise
const user = use(userPromise);
return <div>{user.name}</div>;
}
// Option 2: Generic type (for reusable components)
export function GenericDataDisplay<T>({
data
}: {
data: Promise<T>
}) {
const result = use(data);
return <div>{JSON.stringify(result)}</div>;
}
// Option 3: Unknown type (when type truly unknown)
export function UnknownDataDisplay({
data
}: {
data: Promise<unknown>
}) {
const result = use(data);
return <div>{JSON.stringify(result)}</div>;
}Server Component passing promise:
typescript
// app/profile/page.tsx
import UserProfile from './components/UserProfile';
export default function ProfilePage() {
// Create promise but don't await
const userPromise = fetchUser();
return (
<Suspense fallback={<div>Loading...</div>}>
<UserProfile userPromise={userPromise} />
</Suspense>
);
}typescript
// app/components/UserProfile.tsx
'use client';
import { use } from 'react';
// 重要:使用特定类型、泛型类型或'unknown' - 绝对不要使用'any'
// 选项1:特定类型(已知类型时最佳)
export default function UserProfile({
userPromise
}: {
userPromise: Promise<{ name: string; email: string }>
}) {
// 解包Promise
const user = use(userPromise);
return <div>{user.name}</div>;
}
// 选项2:泛型类型(用于可复用组件)
export function GenericDataDisplay<T>({
data
}: {
data: Promise<T>
}) {
const result = use(data);
return <div>{JSON.stringify(result)}</div>;
}
// 选项3:Unknown类型(类型未知时)
export function UnknownDataDisplay({
data
}: {
data: Promise<unknown>
}) {
const result = use(data);
return <div>{JSON.stringify(result)}</div>;
}Server Component传递Promise:
typescript
// app/profile/page.tsx
import UserProfile from './components/UserProfile';
export default function ProfilePage() {
// 创建Promise但不等待
const userPromise = fetchUser();
return (
<Suspense fallback={<div>加载中...</div>}>
<UserProfile userPromise={userPromise} />
</Suspense>
);
}Benefits of 'use' API
'use' API的优势
- Enables parallel data fetching
- Works with Suspense boundaries
- Allows Server Components to pass promises to Client Components
- Cleaner than prop drilling async data
- 支持并行数据获取
- 与Suspense边界配合使用
- 允许Server Components向Client Components传递Promise
- 比传递异步数据的属性钻取更简洁
Using 'use' with Context
结合Context使用'use'
typescript
'use client';
import { use } from 'react';
import { ThemeContext } from './ThemeContext';
export default function ThemedButton() {
const theme = use(ThemeContext);
return <button className={theme.buttonClass}>Click me</button>;
}typescript
'use client';
import { use } from 'react';
import { ThemeContext } from './ThemeContext';
export default function ThemedButton() {
const theme = use(ThemeContext);
return <button className={theme.buttonClass}>点击我</button>;
}Common Patterns
常见模式
Pattern 1: Server Component Fetches, Client Component Interacts
模式1:Server Component获取数据,Client Component处理交互
typescript
// app/products/page.tsx (Server Component)
import ProductGrid from './ProductGrid';
export default async function ProductsPage() {
const products = await fetchProducts();
// Pass data to Client Component
return <ProductGrid products={products} />;
}
// app/products/ProductGrid.tsx (Client Component)
'use client';
import { useState } from 'react';
export default function ProductGrid({
products
}: {
products: Product[]
}) {
const [filter, setFilter] = useState('all');
const filtered = products.filter(p =>
filter === 'all' || p.category === filter
);
return (
<div>
<select onChange={(e) => setFilter(e.target.value)}>
<option value="all">All</option>
<option value="electronics">Electronics</option>
</select>
{filtered.map(p => <div key={p.id}>{p.name}</div>)}
</div>
);
}typescript
// app/products/page.tsx (Server Component)
import ProductGrid from './ProductGrid';
export default async function ProductsPage() {
const products = await fetchProducts();
// 将数据传递给Client Component
return <ProductGrid products={products} />;
}
// app/products/ProductGrid.tsx (Client Component)
'use client';
import { useState } from 'react';
export default function ProductGrid({
products
}: {
products: Product[]
}) {
const [filter, setFilter] = useState('all');
const filtered = products.filter(p =>
filter === 'all' || p.category === filter
);
return (
<div>
<select onChange={(e) => setFilter(e.target.value)}>
<option value="all">全部</option>
<option value="electronics">电子产品</option>
</select>
{filtered.map(p => <div key={p.id}>{p.name}</div>)}
</div>
);
}Pattern 2: Parallel Data Fetching
模式2:并行数据获取
typescript
// app/dashboard/page.tsx
export default async function Dashboard() {
// Fetch in parallel
const [user, stats, notifications] = await Promise.all([
fetchUser(),
fetchStats(),
fetchNotifications(),
]);
return (
<div>
<UserInfo user={user} />
<Stats data={stats} />
<Notifications items={notifications} />
</div>
);
}typescript
// app/dashboard/page.tsx
export default async function Dashboard() {
// 并行获取
const [user, stats, notifications] = await Promise.all([
fetchUser(),
fetchStats(),
fetchNotifications(),
]);
return (
<div>
<UserInfo user={user} />
<Stats data={stats} />
<Notifications items={notifications} />
</div>
);
}Pattern 3: Streaming with Suspense
模式3:结合Suspense的流式渲染
typescript
// app/page.tsx
import { Suspense } from 'react';
export default function Page() {
return (
<div>
<h1>Dashboard</h1>
<Suspense fallback={<div>Loading stats...</div>}>
<Stats />
</Suspense>
<Suspense fallback={<div>Loading feed...</div>}>
<Feed />
</Suspense>
</div>
);
}
async function Stats() {
const data = await fetchStats(); // Slow query
return <div>{data.total}</div>;
}
async function Feed() {
const items = await fetchFeed(); // Fast query
return <ul>{items.map(i => <li key={i.id}>{i.title}</li>)}</ul>;
}typescript
// app/page.tsx
import { Suspense } from 'react';
export default function Page() {
return (
<div>
<h1>仪表盘</h1>
<Suspense fallback={<div>加载统计数据...</div>}>
<Stats />
</Suspense>
<Suspense fallback={<div>加载动态内容...</div>}>
<Feed />
</Suspense>
</div>
);
}
async function Stats() {
const data = await fetchStats(); // 慢查询
return <div>{data.total}</div>;
}
async function Feed() {
const items = await fetchFeed(); // 快查询
return <ul>{items.map(i => <li key={i.id}>{i.title}</li>)}</ul>;
}Pattern 4: Composition - Server Inside Client
模式4:组合 - Server Component嵌套在Client Component中
You CAN pass Server Components as children to Client Components:
typescript
// app/page.tsx (Server Component)
import ClientWrapper from './ClientWrapper';
import ServerContent from './ServerContent';
export default function Page() {
return (
<ClientWrapper>
{/* Server Component as children */}
<ServerContent />
</ClientWrapper>
);
}
// ClientWrapper.tsx (Client Component)
'use client';
import { useState } from 'react';
export default function ClientWrapper({
children
}: {
children: React.ReactNode
}) {
const [isOpen, setIsOpen] = useState(false);
return (
<div>
<button onClick={() => setIsOpen(!isOpen)}>Toggle</button>
{isOpen && children}
</div>
);
}
// ServerContent.tsx (Server Component)
export default async function ServerContent() {
const data = await fetchData();
return <div>{data.content}</div>;
}可以将Server Component作为子组件传递给Client Component:
typescript
// app/page.tsx (Server Component)
import ClientWrapper from './ClientWrapper';
import ServerContent from './ServerContent';
export default function Page() {
return (
<ClientWrapper>
{/* Server Component作为子组件 */}
<ServerContent />
</ClientWrapper>
);
}
// ClientWrapper.tsx (Client Component)
'use client';
import { useState } from 'react';
export default function ClientWrapper({
children
}: {
children: React.ReactNode
}) {
const [isOpen, setIsOpen] = useState(false);
return (
<div>
<button onClick={() => setIsOpen(!isOpen)}>切换</button>
{isOpen && children}
</div>
);
}
// ServerContent.tsx (Server Component)
export default async function ServerContent() {
const data = await fetchData();
return <div>{data.content}</div>;
}Anti-Patterns to Avoid
需要避免的反模式
Anti-Pattern 1: Using 'use client' Everywhere
反模式1:到处使用'use client'
Wrong:
typescript
// app/components/Header.tsx
'use client'; // Unnecessary!
export default function Header() {
return <header><h1>My App</h1></header>;
}Correct:
typescript
// app/components/Header.tsx
// No directive needed - keep it as Server Component
export default function Header() {
return <header><h1>My App</h1></header>;
}Why: Only use when you actually need client-side features. Static components should remain Server Components to reduce bundle size.
'use client'错误示例:
typescript
// app/components/Header.tsx
'use client'; // 不必要!
export default function Header() {
return <header><h1>我的应用</h1></header>;
}正确示例:
typescript
// app/components/Header.tsx
// 无需指令 - 保持为Server Component
export default function Header() {
return <header><h1>我的应用</h1></header>;
}原因: 仅在实际需要客户端特性时使用。静态组件应保持为Server Component以减少包体积。
'use client'Anti-Pattern 2: Fetching Data in Client Components
反模式2:在Client Component中获取数据
Wrong:
typescript
'use client';
import { useState, useEffect } from 'react';
export default function Products() {
const [products, setProducts] = useState([]);
useEffect(() => {
fetch('/api/products')
.then(r => r.json())
.then(setProducts);
}, []);
return <div>{products.map(p => <div key={p.id}>{p.name}</div>)}</div>;
}Correct:
typescript
// Server Component - no 'use client'
export default async function Products() {
const response = await fetch('https://api.example.com/products');
const products = await response.json();
return <div>{products.map(p => <div key={p.id}>{p.name}</div>)}</div>;
}Why: Server Components can fetch data directly, eliminating loading states and reducing client-side JavaScript.
错误示例:
typescript
'use client';
import { useState, useEffect } from 'react';
export default function Products() {
const [products, setProducts] = useState([]);
useEffect(() => {
fetch('/api/products')
.then(r => r.json())
.then(setProducts);
}, []);
return <div>{products.map(p => <div key={p.id}>{p.name}</div>)}</div>;
}正确示例:
typescript
// Server Component - 无需'use client'
export default async function Products() {
const response = await fetch('https://api.example.com/products');
const products = await response.json();
return <div>{products.map(p => <div key={p.id}>{p.name}</div>)}</div>;
}原因: Server Component可以直接获取数据,消除加载状态并减少客户端JavaScript。
Anti-Pattern 3: Accessing Server APIs in Client Components
反模式3:在Client Component中访问服务器API
Wrong:
typescript
'use client';
import { cookies } from 'next/headers'; // ERROR!
export default function ClientComponent() {
const cookieStore = cookies(); // This will fail
return <div>...</div>;
}Correct:
typescript
// Server Component
import { cookies } from 'next/headers';
import ClientComponent from './ClientComponent';
export default async function ServerComponent() {
const cookieStore = await cookies();
const token = cookieStore.get('token')?.value;
return <ClientComponent token={token} />;
}Why: , , and other server-only APIs can only be used in Server Components.
cookies()headers()错误示例:
typescript
'use client';
import { cookies } from 'next/headers'; // 错误!
export default function ClientComponent() {
const cookieStore = cookies(); // 会失败
return <div>...</div>;
}正确示例:
typescript
// Server Component
import { cookies } from 'next/headers';
import ClientComponent from './ClientComponent';
export default async function ServerComponent() {
const cookieStore = await cookies();
const token = cookieStore.get('token')?.value;
return <ClientComponent token={token} />;
}原因: 、等服务器专属API仅能在Server Component中使用。
cookies()headers()Anti-Pattern 4: Serial Await (Waterfall)
反模式4:串行等待(瀑布流)
Wrong:
typescript
export default async function Page() {
const user = await fetchUser();
const posts = await fetchPosts(); // Waits for user to finish
const comments = await fetchComments(); // Waits for posts to finish
return <div>...</div>;
}Correct:
typescript
export default async function Page() {
// Fetch in parallel
const [user, posts, comments] = await Promise.all([
fetchUser(),
fetchPosts(),
fetchComments(),
]);
return <div>...</div>;
}Why: Parallel fetching reduces total load time significantly.
错误示例:
typescript
export default async function Page() {
const user = await fetchUser();
const posts = await fetchPosts(); // 等待user请求完成
const comments = await fetchComments(); // 等待posts请求完成
return <div>...</div>;
}正确示例:
typescript
export default async function Page() {
// 并行获取
const [user, posts, comments] = await Promise.all([
fetchUser(),
fetchPosts(),
fetchComments(),
]);
return <div>...</div>;
}原因: 并行获取可显著减少总加载时间。
Anti-Pattern 5: Importing Server Component into Client Component
反模式5:将Server Component导入到Client Component中
Wrong:
typescript
// ClientComponent.tsx
'use client';
import ServerComponent from './ServerComponent'; // This makes it a Client Component!
export default function ClientComponent() {
return <div><ServerComponent /></div>;
}Correct:
typescript
// ParentServerComponent.tsx (Server Component)
import ClientComponent from './ClientComponent';
import ServerComponent from './ServerComponent';
export default function ParentServerComponent() {
return (
<ClientComponent>
<ServerComponent />
</ClientComponent>
);
}Why: Importing a Server Component into a Client Component converts it to a Client Component. Pass it as children instead.
错误示例:
typescript
// ClientComponent.tsx
'use client';
import ServerComponent from './ServerComponent'; // 这会将其变为Client Component!
export default function ClientComponent() {
return <div><ServerComponent /></div>;
}正确示例:
typescript
// ParentServerComponent.tsx (Server Component)
import ClientComponent from './ClientComponent';
import ServerComponent from './ServerComponent';
export default function ParentServerComponent() {
return (
<ClientComponent>
<ServerComponent />
</ClientComponent>
);
}原因: 将Server Component导入到Client Component中会将其转换为Client Component。应改为通过子组件传递。
When Client Components ARE Appropriate
适合使用Client Components的场景
Client Components are the correct choice for:
Client Components适用于以下场景:
1. Interactive Forms
1. 交互式表单
typescript
'use client';
import { useState } from 'react';
export default function ContactForm() {
const [email, setEmail] = useState('');
const handleSubmit = (e: React.FormEvent) => {
e.preventDefault();
// Handle submission
};
return (
<form onSubmit={handleSubmit}>
<input
value={email}
onChange={(e) => setEmail(e.target.value)}
/>
<button type="submit">Submit</button>
</form>
);
}typescript
'use client';
import { useState } from 'react';
export default function ContactForm() {
const [email, setEmail] = useState('');
const handleSubmit = (e: React.FormEvent) => {
e.preventDefault();
// 处理提交
};
return (
<form onSubmit={handleSubmit}>
<input
value={email}
onChange={(e) => setEmail(e.target.value)}
/>
<button type="submit">提交</button>
</form>
);
}2. Real-Time Features
2. 实时功能
typescript
'use client';
import { useEffect, useState } from 'react';
export default function LiveChat() {
const [messages, setMessages] = useState([]);
useEffect(() => {
const ws = new WebSocket('wss://chat.example.com');
ws.onmessage = (event) => {
setMessages(prev => [...prev, event.data]);
};
return () => ws.close();
}, []);
return <div>{messages.map((m, i) => <div key={i}>{m}</div>)}</div>;
}typescript
'use client';
import { useEffect, useState } from 'react';
export default function LiveChat() {
const [messages, setMessages] = useState([]);
useEffect(() => {
const ws = new WebSocket('wss://chat.example.com');
ws.onmessage = (event) => {
setMessages(prev => [...prev, event.data]);
};
return () => ws.close();
}, []);
return <div>{messages.map((m, i) => <div key={i}>{m}</div>)}</div>;
}3. Browser-Only Features
3. 浏览器专属功能
typescript
'use client';
import { useState, useEffect } from 'react';
export default function GeolocationDisplay() {
const [location, setLocation] = useState(null);
useEffect(() => {
navigator.geolocation.getCurrentPosition((pos) => {
setLocation({
lat: pos.coords.latitude,
lng: pos.coords.longitude,
});
});
}, []);
return location ? <div>Lat: {location.lat}, Lng: {location.lng}</div> : null;
}typescript
'use client';
import { useState, useEffect } from 'react';
export default function GeolocationDisplay() {
const [location, setLocation] = useState(null);
useEffect(() => {
navigator.geolocation.getCurrentPosition((pos) => {
setLocation({
lat: pos.coords.latitude,
lng: pos.coords.longitude,
});
});
}, []);
return location ? <div>纬度:{location.lat},经度:{location.lng}</div> : null;
}4. Third-Party Libraries Requiring Window
4. 需要Window对象的第三方库
typescript
'use client';
import { useEffect, useState } from 'react';
import confetti from 'canvas-confetti';
export default function CelebrationButton() {
const handleClick = () => {
confetti();
};
return <button onClick={handleClick}>Celebrate!</button>;
}typescript
'use client';
import { useEffect, useState } from 'react';
import confetti from 'canvas-confetti';
export default function CelebrationButton() {
const handleClick = () => {
confetti();
};
return <button onClick={handleClick}>庆祝!</button>;
}5. React Context Providers
5. React Context提供者
typescript
'use client';
import { createContext, useState } from 'react';
export const ThemeContext = createContext();
export function ThemeProvider({ children }) {
const [theme, setTheme] = useState('light');
return (
<ThemeContext.Provider value={{ theme, setTheme }}>
{children}
</ThemeContext.Provider>
);
}typescript
'use client';
import { createContext, useState } from 'react';
export const ThemeContext = createContext();
export function ThemeProvider({ children }) {
const [theme, setTheme] = useState('light');
return (
<ThemeContext.Provider value={{ theme, setTheme }}>
{children}
</ThemeContext.Provider>
);
}Quick Decision Tree
快速决策树
Need interactivity? (onClick, onChange, etc.)
├─ Yes → Client Component ('use client')
└─ No → Continue...
Need React hooks? (useState, useEffect, etc.)
├─ Yes → Client Component ('use client')
└─ No → Continue...
Need browser APIs? (window, localStorage, etc.)
├─ Yes → Client Component ('use client')
└─ No → Continue...
Need to fetch data?
├─ Yes → Server Component (default)
└─ No → Continue...
Need cookies/headers/searchParams?
├─ Yes → Server Component (default)
└─ No → Server Component (default, unless specific need)需要交互吗?(onClick、onChange等)
├─ 是 → Client Component(使用'use client')
└─ 否 → 继续...
需要React钩子吗?(useState、useEffect等)
├─ 是 → Client Component(使用'use client')
└─ 否 → 继续...
需要浏览器API吗?(window、localStorage等)
├─ 是 → Client Component(使用'use client')
└─ 否 → 继续...
需要获取数据吗?
├─ 是 → Server Component(默认)
└─ 否 → 继续...
需要cookies/headers/searchParams吗?
├─ 是 → Server Component(默认)
└─ 否 → Server Component(默认,除非有特定需求)Testing Component Type
测试组件类型
To verify component type:
typescript
// This works = Server Component
export default async function MyComponent() { ... }
// This works = Server Component
import { cookies } from 'next/headers';
// This works = Client Component
'use client';
import { useState } from 'react';
// This fails = Wrong combination
'use client';
import { cookies } from 'next/headers'; // ERROR!验证组件类型:
typescript
// 可以运行 = Server Component
export default async function MyComponent() { ... }
// 可以运行 = Server Component
import { cookies } from 'next/headers';
// 可以运行 = Client Component
'use client';
import { useState } from 'react';
// 会失败 = 错误组合
'use client';
import { cookies } from 'next/headers'; // 错误!Summary
总结
- Default to Server Components - they're faster and more secure
- Use Client Components only when you need interactivity or browser APIs
- Never fetch data in Client Components with useEffect - use Server Components
- Pass promises to Client Components with React 'use' API
- Access cookies/headers/searchParams only in Server Components
- Use composition pattern to mix Server and Client Components
- Fetch in parallel with Promise.all to avoid waterfalls
- 默认使用Server Components - 它们更快、更安全
- 仅在需要交互或浏览器API时使用Client Components
- 永远不要在Client Component中使用useEffect获取数据 - 使用Server Components
- 通过React 'use' API向Client Components传递Promise
- 仅在Server Components中访问cookies/headers/searchParams
- 使用组合模式混合Server和Client Components
- 使用Promise.all并行获取数据以避免瀑布流