React源码|如何在渲染中,创建虚拟DOM对象 ?

一、整体的思路

  1. 整体上通过jsxDev函数,通过传入的参数,返回一个虚拟dom对象。
  2. 明确传入的参数
  3. 明确返回对象的属性
  4. 明确中间的实现步骤
  • 处理key和ref属性,
  • 并将其他属性添加到props对象中

二、认识JSX

1. 在官方文档里这么说的

JSX 是 JavaScript 语法扩展,可以让你在 JavaScript 文件中书写类似 HTML 的标签。虽然还有其它方式可以编写组件,但大部分 React 开发者更喜欢 JSX 的简洁性,并且在大部分代码库中使用它。

2. Javascript 本身不支持JSX

如果在普通的 JavaScript 程序中编写类似于下面的代码, 是会报错的。

javascript 复制代码
const LION = <div>LION</div>

3. 让JavaScript 负责 HTML 的内容!

随着Web的交互性越来越强,逻辑越来越决定页面中的内容。这也是为什么在React中,渲染逻辑和标签共存在一个地方------组件。

4. Babel对代码做了转换

在浏览器或服务器端运行之前,JSX代码需要通过Babel等编译器转换成普通的JavaScript。这个过程中,像<div>这样的JSX元素会被转换成React的createElement函数调用。

jsx 复制代码
let element = <div>
    <div>狮子</div>
    <div>LION<a style={{ color: 'blue' }} href="http://www.baidu.com">http://www.baidu.com</a></div>
</div>
javascript 复制代码
let element = React.createElement("div", null,
    React.createElement("div", null, "狮子"),
    React.createElement("div", null, "LION:", 
        React.createElement("a", { style: { color: 'blue' }, href: "https://www.baidu.com" }, "https://www.baidu.com")
    )
);

三、代码实现

1. 明确输入与输出

  • 传入的参数为 type, config

type是节点的类型,config是参数,maybekey是为了做意外处理

  • 返回的结果:
jsx 复制代码
{
    $$typeof: REACT_ELEMENT_TYPE, // 虚拟dom的类型
    type, // 标签类型
    key, // key
    ref, // ref
    props // 传入参数
}

2. 创建 jsxDEV函数

jsx 复制代码
// 创建一个React元素的函数,处理key和ref属性,并将其他属性添加到props对象中
export function jsxDEV(type, config, maybeKey) {
  // 返回一个新的React元素
  return ReactElement(type, key, ref, props);
}

3. 初始化

jsx 复制代码
let propName;  
const props = {};
let key = null;
let ref = null; 

4. 如果maybeKey参数存在

jsx 复制代码
// 如果maybeKey参数存在,将其赋值给key
if (typeof maybeKey !== 'undefined') {
    key = maybeKey;
}

if (hasValidKey(config)) {
    key = '' + config.key;
}

5. 如果config对象中有ref

jsx 复制代码
// 如果config对象中有ref属性,将其赋值给ref
if (hasValidRef(config)) {
    ref = config.ref;
} 

6. 遍历config对象

jsx 复制代码
// 遍历config对象,将非保留属性添加到props对象中
for (propName in config) {
    if (hasOwnProperty.call(config, propName)
      && !RESERVED_PROPS.hasOwnProperty(propName)
    ) {
      props[propName] = config[propName]
    }
  }

7. 创建对象

jsx 复制代码
// 创建一个React元素(虚拟DOM)
function ReactElement(type, key, ref, props) {
  return {
    $$typeof: REACT_ELEMENT_TYPE,
    type,
    key,
    ref,
    props
  }
}

8. 完整代码

jsx 复制代码
// 创建一个React元素的函数,处理key和ref属性,并将其他属性添加到props对象中
export function jsxDEV(type, config, maybeKey) {
  let propName;
  const props = {};
  let key = null;
  let ref = null; 

  // 如果maybeKey参数存在,将其赋值给key
  if (typeof maybeKey !== 'undefined') {
    key = maybeKey;
  }

  // 检查config对象中是否有ref属性
  if (config.key !== undefined) {
    key = '' + config.key;
  }

  // 如果config对象中有ref属性,将其赋值给ref
  if (hasValidRef(config)) {
    ref = config.ref;
  } 

  // 遍历config对象,将非保留属性添加到props对象中
  for (propName in config) {
    if (hasOwnProperty.call(config, propName)
      && !RESERVED_PROPS.hasOwnProperty(propName)
    ) {
      props[propName] = config[propName]
    }
  }

  // 返回一个新的React元素
  return ReactElement(type, key, ref, props);
}

参考内容:

  1. 慕课网课程:手写 React 高质量源码,迈向高阶开发
  2. React中文网: react.nodejs.cn/
  3. React源码:github.com/facebook/re...
相关推荐
爱学习的程序媛1 小时前
【Web前端】JavaScript设计模式全解析
前端·javascript·设计模式·web
小码哥_常1 小时前
从SharedPreferences到DataStore:Android存储进化之路
前端
老黑1 小时前
开源工具 AIDA:给 AI 辅助开发加一个数据采集层,让 AI 从错误中自动学习(Glama 3A 认证)
前端·react.js·ai·nodejs·cursor·vibe coding·claude code
jessecyj1 小时前
Spring boot整合quartz方法
java·前端·spring boot
苦瓜小生2 小时前
【前端】|【js手撕】经典高频面试题:手写实现function.call、apply、bind
java·前端·javascript
天若有情6732 小时前
前端HTML精讲03:页面性能优化+懒加载,搞定首屏加速
前端·性能优化·html
踩着两条虫2 小时前
AI驱动的Vue3应用开发平台深入探究(十):物料系统之内置组件库
android·前端·vue.js·人工智能·低代码·系统架构·rxjava
swipe2 小时前
AI 应用里的 Memory,不是“保存聊天记录”,而是管理上下文预算
前端·llm·agent
慧一居士3 小时前
nuxt3 项目和nuxt4 项目区别和对比
前端·vue.js
威联通安全存储3 小时前
破除“重前端、轻底层”的数字幻象:如何夯实工业数据的物理底座
前端·python