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...
相关推荐
qq_177767379 分钟前
React Native鸿蒙跨平台实现应用介绍页,实现了应用信息卡片展示、特色功能网格布局、权限/联系信息陈列、评分展示、模态框详情交互等通用场景
javascript·react native·react.js·ecmascript·交互·harmonyos
wuhen_n21 分钟前
JavaScript内存管理与执行上下文
前端·javascript
Hi_kenyon42 分钟前
理解vue中的ref
前端·javascript·vue.js
jin1233222 小时前
基于React Native鸿蒙跨平台地址管理是许多电商、外卖、物流等应用的重要功能模块,实现了地址的添加、编辑、删除和设置默认等功能
javascript·react native·react.js·ecmascript·harmonyos
2501_920931702 小时前
React Native鸿蒙跨平台医疗健康类的血压记录,包括收缩压、舒张压、心率、日期、时间、备注和状态
javascript·react native·react.js·ecmascript·harmonyos
落霞的思绪2 小时前
配置React和React-dom为CDN引入
前端·react.js·前端框架
Hacker_Z&Q2 小时前
CSS 笔记2 (属性)
前端·css·笔记
Anastasiozzzz2 小时前
LeetCode Hot100 295. 数据流的中位数 MedianFinder
java·服务器·前端
橙露3 小时前
React Hooks 深度解析:从基础使用到自定义 Hooks 的封装技巧
javascript·react.js·ecmascript
Exquisite.3 小时前
Nginx
服务器·前端·nginx