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...
相关推荐
街尾杂货店&26 分钟前
css word-spacing属性
前端·css
千叶寻-29 分钟前
正则表达式
前端·javascript·后端·架构·正则表达式·node.js
光影少年6 小时前
angular生态及学习路线
前端·学习·angular.js
无尽夏_8 小时前
HTML5(前端基础)
前端·html·html5
Jagger_8 小时前
敏捷开发流程-精简版
前端·后端
FIN66688 小时前
昂瑞微冲刺科创板:创新驱动,引领射频芯片国产化新征程
前端·安全·前端框架·信息与通信·芯片
GISer_Jing8 小时前
ByteDance——jy真题
前端·javascript·面试
睡美人的小仙女1278 小时前
浏览器为何屏蔽本地文件路径?
前端
真的想不出名儿8 小时前
Vue 中 props 传递数据的坑
前端·javascript·vue.js
FIN66688 小时前
昂瑞微:深耕射频“芯”赛道以硬核实力冲刺科创板大门
前端·人工智能·科技·前端框架·信息与通信·智能