React第六十二节 Router中 createStaticRouter 的使用详解

前言

createStaticRouterReact Router 专为 服务端渲染(SSR) 设计的 API ,用于在服务器端处理路由匹配数据加载。它在构建静态 HTML 响应时替代了客户端的 BrowserRouter,确保 SSR 和客户端 Hydration 的路由状态一致。

一、createStaticRouter 核心用途

  1. 服务端路由匹配 :根据请求 URL 确定渲染的组件
  2. 数据预加载 :执行路由的 loader 函数获取初始数据
  3. 错误处理:捕获渲染过程中的路由级错误
  4. SSR/SSG 支持:生成包含初始数据的静态 HTML

二、createStaticRouter 使用步骤详解(配合 Express 示例)

2.1、 定义路由配置

javascript 复制代码
// src/routes.js
import HomePage from "./pages/Home";
import UserPage from "./pages/User";

export const routes = [
  {
    path: "/",
    loader: () => fetch("/api/data"), // 数据加载函数
    element: <HomePage />,
    errorElement: <ErrorPage /> // 错误边界
  },
  {
    path: "/user/:id",
    loader: ({ params }) => fetch(`/api/users/${params.id}`),
    element: <UserPage />
  }
];

2.2、 服务端路由处理

javascript 复制代码
// server.js
import express from "express";
import { 
  createStaticRouter,
  StaticRouterProvider 
} from "react-router-dom/server";
import { routes } from "./src/routes";

const app = express();

app.use("*", async (req, res) => {
  // 1. 创建请求感知的静态路由
  const router = createStaticRouter(routes, {
    basename: "/app",      // 基础路径
    location: req.originalUrl // 当前请求路径
  });

  // 2. 触发所有匹配路由的 loader
  const promises = router.matches.map(match => 
    match.route.loader?.({ request: req, params: match.params })
  );

  // 3. 等待数据加载完成
  const loaderData = await Promise.all(promises);

  // 4. 将数据注入路由上下文
  const context = {
    loaderData,
    errors: null // 可捕获 loader 错误
  };

  // 5. 渲染为 HTML 字符串
  const html = ReactDOMServer.renderToString(
    <StaticRouterProvider 
      router={router} 
      context={context} 
    />
  );

  // 6. 拼接完整 HTML 响应
  res.send(`
    <html>
      <body>
        <div id="root">${html}</div>
        <script>
          // 注入初始数据供客户端 Hydration 使用
          window.__STATIC_CONTEXT = ${JSON.stringify(context)};
        </script>
      </body>
    </html>
  `);
});

2.3、 客户端 Hydration

javascript 复制代码
// src/client.js
import { hydrateRoot } from "react-dom/client";
import { createBrowserRouter, RouterProvider } from "react-router-dom";
import { routes } from "./routes";

// 复用路由配置
const router = createBrowserRouter(routes, {
  basename: "/app",
  hydrationData: window.__STATIC_CONTEXT // 注入服务端数据
});

hydrateRoot(
  document.getElementById("root"),
  <RouterProvider router={router} />
);

三、createStaticRouter关键配置说明

参数 作用

  1. basename: 应用基础路径 (e.g. /app)
  2. location: 当前请求 URL 对象 (必需)
  3. router.matches: 当前 URL 匹配的路由对象数组
  4. context.loaderData: 存储 loader 返回数据的数组,索引与 router.matches 顺序一致

四、createStaticRouter 错误处理机制

javascript 复制代码
// 在路由配置中添加错误边界
{
  path: "/user/:id",
  element: <UserPage />,
  errorElement: <ErrorLayout />, // 捕获本路由及子路由错误
  loader: async () => {
    const res = await fetchData();
    if (res.status === 404) {
      throw new Response("Not Found", { status: 404 }); // 抛出错误
    }
    return res.json();
  }
}

// 服务端捕获错误
try {
  await Promise.all(promises);
} catch (error) {
  context.errors = error; // 传递到 StaticRouterProvider
}

五、createStaticRouter 最佳实践

  1. 数据序列化 :确保 loader 返回的数据可被序列化为 JSON
  2. 错误类型 :使用 Response 对象抛出 HTTP 错误状态
  3. 路由复用:服务端/客户端使用相同的路由配置对象
  4. 缓存控制 :对静态路由实现 loader 数据缓存

注意createStaticRouter 仅用于服务端环境,客户端应使用 createBrowserRoutercreateMemoryRouter

相关推荐
0思必得01 天前
[Web自动化] Selenium处理动态网页
前端·爬虫·python·selenium·自动化
摘星编程1 天前
React Native鸿蒙版:Drawer抽屉导航实现
react native·react.js·harmonyos
东东5161 天前
智能社区管理系统的设计与实现ssm+vue
前端·javascript·vue.js·毕业设计·毕设
catino1 天前
图片、文件的预览
前端·javascript
2501_920931701 天前
React Native鸿蒙跨平台实现推箱子游戏,完成玩家移动与箱子推动,当所有箱子都被推到目标位置时,玩家获胜
javascript·react native·react.js·游戏·ecmascript·harmonyos
layman05281 天前
webpack5 css-loader:从基础到原理
前端·css·webpack
半桔1 天前
【前端小站】CSS 样式美学:从基础语法到界面精筑的实战宝典
前端·css·html
AI老李1 天前
PostCSS完全指南:功能/配置/插件/SourceMap/AST/插件开发/自定义语法
前端·javascript·postcss
_OP_CHEN1 天前
【前端开发之CSS】(一)初识 CSS:网页化妆术的终极指南,新手也能轻松拿捏页面美化!
前端·css·html·网页开发·样式表·界面美化
啊哈一半醒1 天前
CSS 主流布局
前端·css·css布局·标准流 浮动 定位·flex grid 响应式布局