博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
从 0 到 1 实现 React 系列 —— JSX 和 Virtual DOM
阅读量:5835 次
发布时间:2019-06-18

本文共 3292 字,大约阅读时间需要 10 分钟。

看源码一个痛处是会陷进理不顺主干的困局中,本系列文章在实现一个 (x)react 的同时理顺 React 框架的主干内容(JSX/虚拟DOM/...)

环境准备

项目打包工具选择了 parcel,使用其可以快速地进入项目开发的状态。

此外需要安装以下 babel 插件:

"babel-core": "^6.26.0","babel-preset-env": "^1.6.1","babel-plugin-transform-react-jsx": "^6.24.1"

同时 .babelrc 配置如下:

{    "presets": ["env"],    "plugins": [        // 插件如其名:转化 JSX 语法为定义的形式        ["transform-react-jsx", {            "pragma": "React.createElement"        }]    ]}

JSX 和 虚拟 DOM

const element = (  
hello
world!
)

JSX 是一种语法糖,经过 转换结果如下,可以发现实际上转化成 React.createElement() 的形式:

var element = React.createElement(  "div",  { className: "title" },  "hello",  React.createElement(    "span",    { className: "content" },    "world!"  ));

element, 结果如下:

{  attributes: {className: "title"}  children: ["hello", t] // t 和外层对象相同  key: undefined  nodeName: "div"}

因此,我们得出结论:JSX 语法糖经过 Babel 编译后转换成一种对象,该对象即所谓的虚拟 DOM,使用虚拟 DOM 能让页面进行更为高效的渲染。

我们按照这种思路进行函数的构造:

const React = {  createElement}function createElement(tag, attr, ...child) {  return {    attributes: attr,    children: child,    key: undefined,    nodeName: tag,  }}// 测试const element = (  
hello
world!
)console.log(element) // 打印结果符合预期// {// attributes: {className: "title"}// children: ["hello", t] // t 和外层对象相同// key: undefined// nodeName: "div"// }

虚拟 DOM 转化为真实 DOM

上个小节介绍了 JSX 转化为虚拟 DOM 的过程,这个小节接着来实现将虚拟 DOM 转化为真实 DOM (页面上渲染的是真实 DOM)。

我们知道在 React 中,将虚拟 DOM 转化为真实 DOM 是使用 ReactDOM.render 实现的,使用如下:

ReactDOM.render(  element, // 上文的 element,即虚拟 dom  document.getElementById('root'))

接着来实现 ReactDOM.render 的逻辑:

const ReactDOM = {  render}/** * 将虚拟 DOM 转化为真实 DOM * @param {*} vdom      虚拟 DOM * @param {*} container 需要插入的位置 */function render(vdom, container) {  if (typeof(vdom) === 'string') {    container.innerText = vdom    return  }  const dom = document.createElement(vdom.nodeName)  for (let attr in vdom.attributes) {    setAttribute(dom, attr, vdom.attributes[attr])  }  vdom.children.forEach(vdomChild => render(vdomChild, dom))  container.appendChild(dom)}/** * 给节点设置属性 * @param {*} dom   操作元素 * @param {*} attr  操作元素属性 * @param {*} value 操作元素值 */function setAttribute(dom, attr, value) {  if (attr === 'className') {    attr = 'class'  }  if (attr.match('/on\w+/')) {   // 处理事件的属性:    const eventName = attr.toLowerCase().splice(1)    dom.addEventListener(eventName, value)  } else if (attr === 'style') { // 处理样式的属性:    let styleStr = ''    let standardCss    for (let klass in value) {      standardCss = humpToStandard(klass) // 处理驼峰样式为标准样式      styleStr += `${standardCss}: ${value[klass]};`    }    dom.setAttribute(attr, styleStr)  } else {                       // 其它属性    dom.setAttribute(attr, value)  }}

至此,我们成功将虚拟 DOM 复原为真实 DOM,展示如下:

另外配合热更新,在热更新的时候清空之前的 dom 元素,改动如下:

const ReactDOM = {  render(vdom, container) {    container.innerHTML = null    render(vdom, container)  }}

总结

JSX 经过 babel 编译为 React.createElement() 的形式,其返回结果就是 Virtual DOM,最后通过 ReactDOM.render() 将 Virtual DOM 转化为真实的 DOM 展现在界面上。流程图如下:

思考题

如下是一个 react/preact 的常用组件的写法,那么为什么要 import 一个 React 或者 h 呢?

import React, { Component } from 'react' // react// import { h, Component } from 'preact' // preactclass A extends Component {  render() {    return 
I'm componentA
}}render(, document.body) // 组件的挂载

项目说明

该系列文章会尽可能的分析项目细节,具体的还是以项目实际代码为准。

转载地址:http://juycx.baihongyu.com/

你可能感兴趣的文章
Java并发框架——什么是AQS框架
查看>>
【数据库】
查看>>
WindowManager.LayoutParams 详解
查看>>
find的命令的使用和文件名的后缀
查看>>
Android的Aidl安装方法
查看>>
Linux中rc的含义
查看>>
asp.net怎样在URL中使用中文、空格、特殊字符
查看>>
实现跨交换机VLAN间的通信
查看>>
python例子
查看>>
环境变量(总结)
查看>>
ios之UILabel
查看>>
Java基础之String,StringBuilder,StringBuffer
查看>>
1月9日学习内容整理:爬虫基本原理
查看>>
安卓中数据库的搭建与使用
查看>>
AT3908 Two Integers
查看>>
C++ 0X 新特性实例(比较常用的) (转)
查看>>
node生成自定义命令(yargs/commander)
查看>>
.NET 设计规范--.NET约定、惯用法与模式-2.框架设计基础
查看>>
win7 64位+Oracle 11g 64位下使用 PL/SQL Developer 的解决办法
查看>>
BZOJ1997:[HNOI2010]PLANAR——题解
查看>>