实现自己的mini-React 1(手摸手系列)

写在开头,本系列篇幅较长包含大量代码及思考逻辑,主要实现涉及React中的:

  • Function Component
  • useEffect
  • useState
  • 点击事件处理及props绑定
  • 任务调度器
  • Fiber架构(树结构到链表结构转换)

文章会从main.jsx结合jsx处理后的dom树及源码带入react的大概实现逻辑,先上代码链接(后续会更新优化,方便阅读),文章产出源自@阿崔cxrmini-react活动课程。

Step 1 项目搭建

本文使用vite搭建的js空项目,后续可以直接上jsx语法,结合react项目的入口文件开始反推;

如上图所示,关键代码在最后一行,意在通过ReactDOMcreateRoot传入一个根节点,返回一个render函数处理App组件;可以猜想到是App组件在通过jsx转换后给到render函数渲染vDom,最后挂载到根节点上。

拆分任务

  • 先上js语法,从最基础的dom创建、vDom生成
  • 修改函数名及层级结构,对齐react
js 复制代码
// 创建 vdom

function createTextNode(text){
    return {
        type: 'TEXT_ELEMENT',
        props: {
            nodeValue: text,
            children: []
        }
    }
}
function createElement(type, props, ...children){
    return {
        type,
        props: {
            ...props,
            children: children.map(el => typeof el === 'string'?createTextNode(el):el)
        }
    }
}
/**
 * 
 * @param {创建元素} el 
 * @param {容器元素} container 
 */
function myRender(el,container) {
    // 创建元素
    const dom = el.type === 'TEXT_ELEMENT' 
    ? document.createTextNode('')
    : document.createElement(el.type)
    // 处理props
    Object.keys(el.props).map(key =>{
        if(key !== 'children'){
            // 属性赋值
            dom[key] = el['props'][key]
        }
    })
        // 处理children
        el['props']['children'].map(child =>{
            // 插入父节点
            myRender(child, dom)
        })
    container.appendChild(dom)
}

const root = document.querySelector('#root')
// root.appendChild(dom)

// ?1 优化
const App = createElement('div',{id:'app'},'test 03')
// myRender(App,root)

const ReactDOM = {
    createRoot(container){
        return {
            render(el){
                myRender(el,container)
            }
        }
    }
}

ReactDOM.createRoot(root).render(App)

vDom创建与函数名对齐

根据官网的const element = createElement(type, props, ...children),我们知道创建dom需要传递的数据及类型,而后区分节点类型是文本节点还是其他dom节点。

关于vDom的数据结构,其中props嵌套children,可以拉平也可以嵌套,嵌套是为了和react对齐。

文件层级划分及拆分React.js、ReactDom.js

拆分结果ReactDom.js图:

App.js 代码结构:

js 复制代码
import React from './core/React.js';
const App = createElement('div',{id:'app'},'test 03')
export default App

main.js 的代码结构图:

React.js 代码结构,修改myRender=>render对齐react:

js 复制代码
/ 创建 vdom

function createTextNode(text){
    return {
        type: 'TEXT_ELEMENT',
        props: {
            nodeValue: text,
            children: []
        }
    }
}
function createElement(type, props, ...children){
    return {
        type,
        props: {
            ...props,
            children: children.map(el => typeof el === 'string'?createTextNode(el):el)
        }
    }
}
/**
 * 
 * @param {创建元素} el 
 * @param {容器元素} container 
 */
function render(el,container) {
    // 创建元素
    const dom = el.type === 'TEXT_ELEMENT' 
    ? document.createTextNode('')
    : document.createElement(el.type)
    // 处理props
    Object.keys(el.props).map(key =>{
        if(key !== 'children'){
            // 属性赋值
            dom[key] = el['props'][key]
        }
    })
        // 处理children
        el['props']['children'].map(child =>{
            // 插入父节点
            myRender(child, dom)
        })
    container.appendChild(dom)
}
const React = {
 render,
 createElement
}

到此第一步的工作已经完成,mini-react的形有了,下一步丰富内在。

相关推荐
霍理迪1 小时前
CSS——背景样式以及雪碧图、渐变
前端·css
jump_jump5 小时前
基于 Squoosh WASM 的浏览器端图片转换库
前端·javascript·性能优化
小二·8 小时前
前端监控体系完全指南:从错误捕获到用户行为分析(Vue 3 + Sentry + Web Vitals)
前端·vue.js·sentry
阿珊和她的猫9 小时前
`require` 与 `import` 的区别剖析
前端·webpack
谎言西西里10 小时前
零基础 Coze + 前端 Vue3 边玩边开发:宠物冰球运动员生成器
前端·coze
努力的小郑10 小时前
2025年度总结:当我在 Cursor 里敲下 Tab 的那一刻,我知道时代变了
前端·后端·ai编程
GIS之路10 小时前
GDAL 实现数据空间查询
前端
OEC小胖胖10 小时前
01|从 Monorepo 到发布产物:React 仓库全景与构建链路
前端·react.js·前端框架
2501_9447114311 小时前
构建 React Todo 应用:组件通信与状态管理的最佳实践
前端·javascript·react.js
困惑阿三11 小时前
2025 前端技术全景图:从“夯”到“拉”排行榜
前端·javascript·程序人生·react.js·vue·学习方法