Local MCP Server
在Termux中运行本地MCP服务器,支持Ollama模型的文件读取与命令执行。
提供React与Next.js的性能优化模式,涵盖数据获取、打包大小、组件渲染等关键领域。
openclaw skills install @wpank/react-performance命令、参数、文件名以原文为准
React 与 Next.js 应用的性能优化指南。涵盖 7 个优先级分类的优化模式,按影响程度排序。详细示例见 references/。
| # | 分类 | 影响程度 |
|---|---|---|
| 1 | 异步 / 水瀑布 | 严重 |
| 2 | 打包体积 | 严重 |
| 3 | 服务端组件 | 高 |
| 4 | 重新渲染 | 中 |
| 5 | 渲染 | 中 |
| 6 | 客户端数据 | 中 |
| 7 | JavaScript 性能 | 低-中 |
npx clawhub@latest install react-performance顺序的 await 是 React 应用中最严重的性能问题。
// 错误 — 顺序执行,需 3 次网络请求
const user = await fetchUser()
const posts = await fetchPosts()
const comments = await fetchComments()
// 正确 — 并行执行,仅需 1 次网络请求
const [user, posts, comments] = await Promise.all([
fetchUser(), fetchPosts(), fetchComments(),
])await 直到真正需要时将 await 移入实际使用值的分支中。
// 错误 — 两个分支都阻塞了等待
async function handle(userId: string, skip: boolean) {
const data = await fetchUserData(userId)
if (skip) return { skipped: true } // 仍需等待
return process(data)
}
// 正确 — 仅在需要时才阻塞
async function handle(userId: string, skip: boolean) {
if (skip) return { skipped: true } // 立即返回
return process(await fetchUserData(userId))
}让布局立即显示,而依赖数据的部分独立加载。
// 错误 — 整个页面被阻塞
async function Page() {
const data = await fetchData()
return <div><Sidebar /><Header /><DataDisplay data={data} /><Footer /></div>
}
// 正确 — 布局立即渲染,数据流式加载
function Page() {
return (
<div>
<Sidebar /><Header />
<Suspense fallback={<Skeleton />}><DataDisplay /></Suspense>
<Footer />
</div>
)
}
async function DataDisplay() {
const data = await fetchData()
return <div>{data.content}</div>
}使用 use() 在组件间共享一个 Promise,避免重复请求。
barrel 文件会加载数千个未使用的模块。直接导入可节省 200–800ms。
// 错误 — 加载 1,583 个模块
import { Check, X, Menu } from 'lucide-react'
// 正确 — 仅加载 3 个模块
import Check from 'lucide-react/dist/esm/icons/check'
import X from 'lucide-react/dist/esm/icons/x'
import Menu from 'lucide-react/dist/esm/icons/menu'Next.js 13.5+:在配置中启用 experimental.optimizePackageImports。
常见受影响库:lucide-react、@mui/material、react-icons、@radix-ui、lodash、date-fns。
import dynamic from 'next/dynamic'
const MonacoEditor = dynamic(
() => import('./monaco-editor').then((m) => m.MonacoEditor),
{ ssr: false }
)分析工具、日志记录、错误追踪等,在客户端 hydration 后再加载,使用 dynamic() 和 { ssr: false }。
const preload = () => { void import('./monaco-editor') }
<button onMouseEnter={preload} onFocus={preload} onClick={onClick}>打开编辑器</button>仅传递客户端实际使用的字段。
// 错误 — 序列化全部 50 个用户字段
return <Profile user={user} />
// 正确 — 仅序列化 1 个字段
return <Profile name={user.name} />RSC 在组件树内按顺序执行。通过结构调整实现并行。
// 错误 — 侧边栏需等待头部数据获取完成
export default async function Page() {
const header = await fetchHeader()
return <div><div>{header}</div><Sidebar /></div>
}
// 正确 — 兄弟异步组件同时发起请求
async function Header() { return <div>{await fetchHeader()}</div> }
async function Sidebar() { return <nav>{(await fetchSidebarItems()).map(renderItem)}</nav> }
export default function Page() { return <div><Header /><Sidebar /></div> }React.cache() 实现每请求去重import { cache } from 'react'
export const getCurrentUser = cache(async () => {
const session = await auth()
if (!session?.user?.id) return null
return await db.user.findUnique({ where: { id: session.user.id } })
})使用基础类型参数(而非内联对象)—— React.cache() 使用 Object.is 判断缓存键。Next.js 会自动对 fetch 去重,但数据库查询、认证检查和计算逻辑仍需手动使用 React.cache()。
after() 处理非阻塞操作import { after } from 'next/server'
export async function POST(request: Request) {
await updateDatabase(request)
after(async () => { logUserAction({ userAgent: request.headers.get('user-agent') }) })
return Response.json({ status: 'success' })
}// 错误 — 冗余状态 + 副作用
const [fullName, setFullName] = useState('')
useEffect(() => { setFullName(first + ' ' + last) }, [first, last])
// 正确 — 直接在渲染中推导
const fullName = first + ' ' + last// 错误 — 每次 items 变化都会重新创建
const addItem = useCallback((item: Item) => {
setItems([...items, item])
}, [items])
// 正确 — 稳定,始终获取最新状态
const addItem = useCallback((item: Item) => {
setItems((curr) => [...curr, item])
}, [])如果仅在回调中读取动态状态,请避免订阅。
// 错误 — 每次 searchParams 变化都会重新渲染
const searchParams = useSearchParams()
const handleShare = () => shareChat(chatId, { ref: searchParams.get('ref') })
// 正确 — 按需读取
const handleShare = () => {
const ref = new URLSearchParams(window.location.search).get('ref')
shareChat(chatId, { ref })
}// 错误 — 每次渲染都执行 JSON.parse
const [settings] = useState(JSON.parse(localStorage.getItem('s') || '{}'))
// 正确 — 仅执行一次
const [settings] = useState(() => JSON.parse(localStorage.getItem('s') || '{}'))// 错误 — 每像素变化都会重新渲染
const width = useWindowWidth(); const isMobile = width < 768
// 正确 — 仅在布尔值翻转时重新渲染
const isMobile = useMediaQuery('(max-width: 767px)')// 错误 — 滚动时阻塞 UI
const handler = () => setScrollY(window.scrollY)
// 正确 — 非阻塞
const handler = () => startTransition(() => setScrollY(window.scrollY))const UserAvatar = memo(function UserAvatar({ user }: { user: User }) {
const id = useMemo(() => computeAvatarId(user), [user])
return <Avatar id={id} />
})
function Profile({ user, loading }: Props) {
if (loading) return <Skeleton />
return <div><UserAvatar user={user} /></div>
}注:React Compiler 会自动处理手动
memo()/useMemo()的需求,无需额外添加。
content-visibility对于 1000 项的列表,浏览器可跳过约 990 个不可见元素(初始渲染快 10 倍)。
.list-item { content-visibility: auto; contain-intrinsic-size: 0 80px; }避免重复创建,尤其适用于大型 SVG 节点。React Compiler 会自动完成此优化。
const skeleton = <div className="skeleton" />
function Container() { return <div>{loading && skeleton}</div> }// 错误 — 每个实例独立发起请求
useEffect(() => { fetch('/api/users').then(r => r.json()).then(setUsers) }, [])
// 正确 — 多个实例共享同一请求
const { data: users } = useSWR('/api/users', fetcher)// 错误 — 时间复杂度 O(n)
items.filter(i => allowed.includes(i.id))
// 正确 — 时间复杂度 O(1)
const allowedSet = new Set(allowed)
items.filter(i => allowedSet.has(i.id))// 错误 — 执行 3 次遍历
const a = users.filter(u => u.isAdmin)
const t = users.filter(u => u.isTester)
// 正确 — 仅执行 1 次遍历
const a: User[] = [], t: User[] = []
for (const u of users) { if (u.isAdmin) a.push(u); if (u.isTester) t.push(u) }此外: 使用提前返回、在循环中缓存属性访问、将正则表达式移出循环、在热点路径中优先使用 for...of。
content-visibility(5)已收录 3 个 Skill