实现简易 React SSR 框架

实现简易 React SSR 框架

框架依赖

  • express
  • react
  • react-dom

创建一个 node server

使用 express 创建一个 node server

js 复制代码
import express from 'express'
import handlerSsr from './ssr'

const app = express()
const port = process.env.PORT || 3000

app.use(express.static('public'))

app.get('*' , (req, res) => {
  // 处理 SSR 逻辑
  return handlerSsr(req, res)
})

app.listen(port, () => {
  console.log(`Server started on http://localhost:${port}`)
});

创建一个根组件

js 复制代码
// client/App.js
import React from 'react';

function App(props) {
    return <div>Hello World!</div>;
}

export default App;

处理 SSR 逻辑

主要分为以下几个步骤:

  1. 解析请求:确定当前请求的URL和任何查询参数。
  2. 获取数据:对于动态内容,您可能需要从数据库或外部API获取数据。
  3. 渲染React组件树:使用收集到的数据渲染您的React组件树。
  4. 序列化HTML:将渲染后的React组件转换为静态HTML字符串。
  5. 发送响应:将最终的HTML字符串发送回浏览器。
js 复制代码
import App from './App'

export const handlerSsr = (req, res) => {
  // 处理请求
  const url = req.path

  const handleRequest = async () => {
      const data = await getDataPage(url);

      if (!data) {
          res.statusCode = 404;
          res.end('Not Found');
          return;
      }

      const html = renderPage(data);
      res.write(html);
      res.end();
  };

  // 获取数据(在真实场景中,这部分很可能是异步的)
  const getDataPage = async (url) => {
      // 模拟从数据库或API获取数据的行为
      switch (url) {
          case '/':
              return { title: 'Home', message: 'Welcome home!' };
          case '/about':
              return { title: 'About Us' };
          default:
              return null;
      }
  };

  // 渲染React组件树
  const renderPage = (data) => {
      const renderedContent = ReactDOMServer.renderToString(
          <App {...data} /> // 将数据作为props传递给App组件
      );

      return `
        <!doctype html>
        <html>
            <head>
                <meta charset=utf-8/>
                <title>${data.title}</title>
            </head>
            <body>
                <div id="root">${renderedContent}</div>
                <script type="text/javascript" src="index_bundle.js"></script>
            </body>
        </html>
    `;
  };

  handleRequest()
      .catch((err) => {
          console.error(err.stack);
          res.status(500).send('Something broke!');
      });

}
相关推荐
UXbot3 小时前
一人独立交付 UI + 前端:AI 驱动 UI 设计工具的五大功能模块深度评测
前端·低代码·ui·设计模式·交互
kobesdu3 小时前
【ROS2实战笔记-19】ROS2 生命周期节点的启动顺序、状态转换陷阱与热备方案
java·前端·笔记·机器人·ros·ros2
诚实可靠王大锤3 小时前
React Native 输入框与按钮焦点冲突解决方案(rn版本0.70.3)
前端·javascript·react native·react.js
kyriewen3 小时前
测试妹子让我写单测,我偷偷用AI一天干完一周的活
前端·chatgpt·cursor
2601_957780843 小时前
Claude Code 2026年最新部署指南:从环境搭建到技能扩展
前端·人工智能·ai编程·claude
zhangfeng11334 小时前
workbuddy 专家 “前端开发师” 结合nvidia-mistral-small-4-119b-2603 项目计划-前端界面开发.md
前端·人工智能·免费
IT_陈寒6 小时前
为什么Java的Stream并行处理反而变慢了?
前端·人工智能·后端
NiceCloud喜云6 小时前
IntelliJ IDEA 保姆级安装 + ClaudeAPI 配置教程
java·开发语言·前端·ide·chrome·docker·intellij-idea
zithern_juejin6 小时前
Date/RegExp/Error/ArrayBuffer
javascript
zenRRan7 小时前
Karpathy公开附议:AI Agent 的输出格式,正在从 Markdown 走向 HTML
前端·html