Skip to content

React 18 新特性预览:并发特性正式来了

React 18 Alpha 已经发布,可以试用了。等了两年的 Concurrent 特性终于要正式了。

并发渲染(Concurrent Rendering)

React 18 的核心变化:渲染变成可以被打断、恢复的过程。

javascript
// React 17:同步渲染
import ReactDOM from "react-dom";
ReactDOM.render(<App />, document.getElementById("root"));

// React 18:并发渲染
import { createRoot } from "react-dom/client";
const root = createRoot(document.getElementById("root"));
root.render(<App />);

只是这一行变化,但开启了很多并发特性。

Automatic Batching(自动批处理)

javascript
// React 17:只有 React 事件处理中才批处理
function handleClick() {
  setCount((c) => c + 1); // 单独渲染
  setName("Alice"); // 单独渲染
  // 两次 setState → 两次渲染
}

// Promise/setTimeout 中不批处理
setTimeout(() => {
  setCount((c) => c + 1); // 渲染 1 次
  setName("Alice"); // 渲染 1 次
  // 两次 setState → 两次渲染!
});

// React 18:所有地方都自动批处理
setTimeout(() => {
  setCount((c) => c + 1);
  setName("Alice");
  // 只渲染 1 次!
});

不需要任何代码改动,性能自动提升。

useTransition

javascript
import { useState, useTransition } from "react";

function SearchPage() {
  const [query, setQuery] = useState("");
  const [results, setResults] = useState([]);
  const [isPending, startTransition] = useTransition();

  function handleSearch(e) {
    // 紧急更新:输入框立即响应
    setQuery(e.target.value);

    // 非紧急(可被打断的)更新
    startTransition(() => {
      const newResults = searchData(e.target.value);
      setResults(newResults);
    });
  }

  return (
    <div>
      <input value={query} onChange={handleSearch} />
      {isPending && <Spinner />}
      <ResultList results={results} />
    </div>
  );
}

用户输入始终流畅,搜索结果更新不会阻塞输入。

useDeferredValue

javascript
import { useState, useDeferredValue } from "react";

function App() {
  const [text, setText] = useState("");
  const deferredText = useDeferredValue(text);

  return (
    <>
      <input value={text} onChange={(e) => setText(e.target.value)} />
      {/* 使用延迟值,不会阻塞 input 响应 */}
      <HeavyList text={deferredText} />
    </>
  );
}

Suspense 改进

javascript
// React 18:Suspense 支持服务端渲染(SSR)
// 页面可以分段流式渲染,不需要等所有数据准备好

function ProfilePage() {
  return (
    <div>
      <Suspense fallback={<HeaderSkeleton />}>
        <Header />
      </Suspense>

      <Suspense fallback={<PostsSkeleton />}>
        <Posts /> {/* 可以独立加载,不需要等 Header */}
      </Suspense>

      <Suspense fallback={<SidebarSkeleton />}>
        <Sidebar />
      </Suspense>
    </div>
  );
}
// 流式 SSR:先发送 HTML 骨架,数据准备好了再流式填充

useId

javascript
import { useId } from 'react'

// 生成唯一 ID(服务端客户端一致,解决 SSR hydration 不匹配问题)
function FormField({ label, type }) {
  const id = useId()

  return (
    <div>
      <label htmlFor={id}>{label}</label>
      <input id={id} type={type} />
    </div>
  )
}

// ✅ 多次使用同一组件,ID 不会冲突
<FormField label="姓名" type="text" />    // id: :r0:
<FormField label="邮箱" type="email" />   // id: :r1:

升级注意事项

javascript
// createRoot 替换 ReactDOM.render(必须的)
// 但在 React 18 中,ReactDOM.render 仍然可用(兼容模式,不开启并发特性)

// 如果用了 useEffect 的副作用,React 18 StrictMode 会执行两次
// 用于检测不纯的副作用(开发模式)
// 生产模式不受影响

小结

  • Automatic Batching 是 React 18 最实际的改进,不需要改代码
  • useTransition:区分紧急/非紧急更新,保持 UI 响应
  • Suspense SSR:流式渲染,用户更快看到内容
  • useId:SSR 和 CSR 一致的唯一 ID
  • 正式版预计 2022 年初发布

MIT Licensed