从零到一:打造你的Mini-React (一)

前段时间,参加了崔哥的活动写一个mini-react,现在来对整个项目进行复盘和总结。

简介

在本篇文章中,我们将一起探索如何从零构建一个迷你版的React框架。通过这个过程,不仅能够加深对React内部机制的理解,还能学习到虚拟DOM、组件生命周期和状态管理等核心概念的实际应用。这篇文章旨在深入浅出地理解React工作原理。

问题清单
Q1:怎么处理jsx?
[Q2: 如果需要渲染数字,怎么处理?](#Q2: 如果需要渲染数字,怎么处理? "#eVH0V")


步骤拆分

简单渲染

我们先来看下react文档的案例

jsx 复制代码
import { createRoot } from 'react-dom/client';

// Clear the existing HTML content
document.body.innerHTML = '<div id="app"></div>';

// Render your React component instead
const root = createRoot(document.getElementById('app'));
root.render(<h1>Hello, world</h1>);

如果我们使用原生JS,可以得到:

jsx 复制代码
const app = document.createElement('div')
app.id = 'app'

const h1 = document.createElement('h1')
h1.textContent = 'Hello world'

app.appendChild(h1)

document.body.append(app)

我们能否用一种数据结构来描述整个结构树,并对渲染函数进一步抽象

  • 虚拟DOM(Virtual DOM)是对真实DOM的抽象表示,并且附加了额外的属性以优化React的更新过程。
jsx 复制代码
// vdom
const vdom = {
  type: 'div',
  props: {
    id: 'app',
    children: [
      {
        type: 'h1',
        props: {
          children: 'Hello world'
        }
      }
    ]
  }
}
jsx 复制代码
// App.jsx
// 这行一定要加上,vite编译时会转为React.createElement使用
import React from "./core/React"

function App() {
  return <h1>Hello, world</h1>
}

export default App

// main.js
import ReactDom from './core/ReactDOM';
import App from './App.jsx';


const root = ReactDom.createRoot(document.querySelector('#root'))
root.render(App)
jsx 复制代码
// React.js
function createElement(type, props, ...children) {
  return {
    type,
    props: {
      ...props,
      children: children.map((child) => {
        const isTextNode = typeof child === 'string'
        return isTextNode ? createTextNode(child) : child
      })
    }
  }
}
function createTextNode(text) {
  return {
    type: 'TEXT_ELEMENT',
    props: {
      nodeValue: text,
      children: []
    }
  }
}

function render(el, container) {
  // 处理函数组件
  if (typeof el === 'function') el = el()
  const dom =
    el.type === 'TEXT_ELEMENT'
      ? document.createTextNode('')
      : document.createElement(el.type)
  const isProperty = (key) => key !== 'children'
  Object.keys(el.props)
    .filter(isProperty)
    .forEach((name) => {
      dom[name] = el.props[name]
    })
  el.props.children.forEach((child) => {
    render(child, dom)
  })
  container.append(dom)
}

const React = {
  render,
  createElement
}

export default React
jsx 复制代码
// ReactDOM
import React from "./React.js";

// 返回一个对象包含render函数
const ReactDOM = {
  createRoot(container) {
    return {
      render(App) {
        React.render(App, container);
      },
    };
  },
};

export default ReactDOM;

Q1:怎么处理jsx

Vite 通过内置插件 @vitejs/plugin-react 来处理 JSX 文件。这个插件的具体作用是在构建阶段将 JSX 代码转换为 React.createElement 调用的形式。这样的转换使得浏览器可以执行转换后的代码,因为浏览器不能直接理解 JSX 语法。

至此,我们已经完成了将元素渲染到页面中

Q2: 如果需要渲染数字,怎么处理?

jsx 复制代码
function App() {
  return (
    <div>
      <h1>Hello, world!</h1>
      <h2>{123}</h2>
    </div>
  )
}

在我们的实现中,元素的创建是通过createElement函数来处理的。为了提高这一过程的健壮性和功能性,我们在现有的基础上添加对数字类型的处理。当检测到数字类型时,系统会自动将其转换为TextNode。

jsx 复制代码
function createElement(type, props, ...children) {
  return {
    type,
    props: {
      ...props,
      children: children.map((child) => {
        // number
        const isTextNode = typeof child === 'string'|| typeof child === 'number'
        return isTextNode ? createTextNode(child) : child
      })
    }
  }
}

总结

在本章节中,我们对从零开始构建一个迷你React框架的基本渲染功能进行了深入研究。我们实现了创建虚拟DOM的过程以及将虚拟DOM转换成实际DOM节点的机制------通常称为渲染(render)方法。此外,我们还阐明了JSX解析的基本原理。下一节,我们将实现批量更新的功能。

相关推荐
深蓝海拓28 分钟前
基于深度学习的视觉检测小项目(十一) 动态样式表的实践
前端·python·pyqt
ChoSeitaku7 小时前
No.1|Godot|俄罗斯方块复刻|棋盘和初始方块的设置
java·前端·godot
生信天地8 小时前
jQuery:前端开发的高效利器
前端·jquery
牛奶皮子8 小时前
vue3Class 与 Style 绑定
前端·javascript·vue.js
傻小胖8 小时前
React setState详细使用总结
前端·react.js·前端框架
网络安全Jack8 小时前
[CTF/网络安全] 攻防世界 Web_php_unserialize 解题详析
前端·web安全·php
宿命小人9 小时前
Electron使用记录
前端·javascript·electron
傻小胖9 小时前
React Error Boundary 错误边界限制
前端·react.js·前端框架
夫琅禾费米线9 小时前
react全局状态管理——redux和zustand,及其区别
前端·javascript·react.js