从零实现 React v18,但 WASM 版 - [1] 项目框架搭建

模仿 big-react,使用 Rust 和 WebAssembly,从 0 到 1 实现从零实现 React v18 的核心功能。深入理解 React 源码的同时,还锻炼了 Rust 的技能,简直赢麻了!

代码地址:github.com/ParadeTo/bi...

本文对应 tag:v1

工具

Rust:一种安全、高效和现代的编程语言(省略一万字)。按照官网所示开发即可。

wasm-pack:构建、测试、和发布 Rust WebAssembly 的一站式工具。

cargo-generate:通过已有的 Git 仓库作为模版,快速创建 Rust 项目。

更多介绍可以参考 Rust 和 WebAssembly 入门教程

其他前端常用的工具自行安装即可,就不赘述了。

项目结构

首先,我们搭建起如下的项目结构:

js 复制代码
.
├── Cargo.toml
├── package.json
├── examples
│   └── hello-world // vite 初始化的 react 项目
├── packages
│   ├── react // 通过 cargo generate --git https://github.com/rustwasm/wasm-pack-template 生成的 WASM 项目
│   ├── react-dom // 通过 cargo generate --git https://github.com/rustwasm/wasm-pack-template 生成的 WASM 项目
│   ├── react-reconciler // cargo new 生成的普通 rust 项目
│   └── shared // cargo new 生成的普通 rust 项目

Cargo.toml 如下所示,类似于前端中常说的 monorepo 架构。

ini 复制代码
[workspace]

members = [
    "packages/react",
    "packages/react-dom",
    "packages/react-reconciler",
    "packages/shared"
]

因为 react 和 react-dom 会导出方法供 JS 侧进行调用,所以需要通过 cargo generate --git https://github.com/rustwasm/wasm-pack-template 创建 WASM 项目,另外两个使用 cargon new 创建普通的 Rust 项目即可。

构建调试环境

我们先删掉 hello-world/src/main.tsx 中的代码,写一个非常简单的例子:

ts 复制代码
import {createRoot} from 'react-dom'

const comp = <div>hello world</div>
console.log(comp)
console.log(createRoot(document.getElementById('root')))

在开发环境运行起来,在浏览器调试窗口中可以看到编译后的代码是这样的:

现在我们的目标是让 hello-world 里面使用的是我们当前正在开发的 React,要让上述代码成功运行,我们需要做以下几步:

  1. 修改 hello-world 项目下的 package.json
json 复制代码
    "react": "file://../../packages/react/pkg/react",
    "react-dom": "file://../../packages/react-dom/pkg/react-dom",
  1. 在根目录下的 package.json 添加打包命令,使用 wasm-pack 把 react 和 react-dom 打包成 WASM:
json 复制代码
  "scripts": {
    "build:react": "wasm-pack build packages/react --out-dir pkg/react --out-name jsx-dev-runtime",
    "build:react-dom": "wasm-pack build packages/react-dom --out-dir pkg/react-dom --out-name index",
    "build": "npm run build:react && npm run build:react-dom"
  },
  1. 分别在 react 和 react-dom 中的 lib.rs 添加如下代码:
rust 复制代码
// react/src/lib.rs
use wasm_bindgen::prelude::*;

#[wasm_bindgen]
pub fn jsxDEV() -> String {
    "hello world".to_string()
}
rust 复制代码
// react-dom/src/lib.rs
use wasm_bindgen::prelude::*;

#[wasm_bindgen]
pub fn createRoot(container: &JsValue) -> JsValue {
    JsValue::null()
}

由于 Rust 中的命名风格一般是下划线风格的,所以最好改成这样:

rust 复制代码
// react/src/lib.rs
use wasm_bindgen::prelude::*;

#[wasm_bindgen(js_name = jsxDev)]
pub fn jsx_dev() -> String {
    "hello world".to_string()
}
rust 复制代码
// react-dom/src/lib.rs
use wasm_bindgen::prelude::*;

#[wasm_bindgen(js_name = createRoot)]
pub fn create_root(container: &JsValue) -> JsValue {
    JsValue::null()
}
  1. 在根目录下运行 npm run build,然后在 hello-world 目录下运行 pnpm installnpm run dev,在浏览器中打开页面,可以看到如下输出:

这说明在 JS 侧成功调用了 WASM 导出的方法,调试环境搭建好了。不过有点麻烦的是,如果修改了代码,需要重新执行步骤 4。

下一篇我们来实现 jsx_dev 的功能,敬请期待。

相关推荐
范特西林7 小时前
一次 to_bits() 引发的 Rust 与 C++ 底层思考
rust
codingWhat8 小时前
手撸一个「能打」的 React Table 组件
前端·javascript·react.js
bluceli8 小时前
WebAssembly实战指南:将高性能计算带入浏览器
前端·webassembly
程序员ys15 小时前
前端权限控制设计
前端·vue.js·react.js
不会敲代码116 小时前
从零开始用 TypeScript + React 打造类型安全的 Todo 应用
前端·react.js·typescript
冬奇Lab1 天前
一天一个开源项目(第42篇):OpenFang - 用 Rust 构建的 Agent 操作系统,16 层安全与 7 个自主 Hands
人工智能·rust·开源
小时前端1 天前
React性能优化的完整方法论,附赠大厂面试通关技巧
前端·react.js
阿慧勇闯大前端1 天前
在AI时代,再去了解react19新特性还有用吗? 最近总有朋友问我:“现在AI写代码这么厉害了,我写个需求丢给ChatGPT,几秒钟就生成一堆组件,还学新特
前端·react.js
量子位1 天前
Transformer论文作者重造龙虾,Rust搓出钢铁版,告别OpenClaw裸奔漏洞
rust·openai·ai编程
喵爱吃鱼2 天前
关于我明明用了ref还是陷入React闭包陷阱
前端·react.js