前端战五渣学React——JSX & React.createElement() & React.ReactElement()源码

image

最近《一拳超人》动画更新第二季了,感觉打斗场面没有第一季那么烧钱了,但是剧情还挺好看的,就找了漫画来看。琦玉老师真的厉害!!!打谁都一拳,就喜欢看老师一拳把那些上来就吹牛逼的反派打的稀烂,专治各种不服!!!

招聘AD

阿里巴巴集团核心前端岗位

薪资25到50

一年一般至少16个月工资

有意者微信联系:Dell-JS

正文

三大民工框架

说到现在的前端,各种招聘JD上都会写

“对主流框架(React/Vue/Angular)有了解,至少深入了解一种”

或者是

“精通MV框架(React/Vue/Angular),至少熟练使用一种,有大型项目经验”*

从中我们可以看出现在前端在工作中使用的框架几乎形成了三足鼎立之势,形如当初的“三大民工漫画”——《海贼王》、《火影忍者》以及《死神》,而Angular又类似《死神》一样,国内人气低迷(我只是从招聘信息来看的。。。angular布道者勿喷)。而React凭借自己的灵活性和vue凭借简单好上手的优势,平分秋色。这回就来主要讲一讲React的一大核心概念——JSX,以及对应的React.createElement()这个方法的源码阅读。

本文阅读需要具备以下知识储备:

  1. JavaScript基本语法,用js开发过项目最好
  2. 最好使用过react,没用过的😅可能。。。

JSX

(了解的可以直接跳到下一节看React.createElement()的源码)

话不多说,让我们来实现一个功能:

创建一个div标签,class名为“title”,内容为“你好 前端战五渣”

看下面的代码⬇️

<!DOCTYPE html>
<html lang="en">
<head>
  <meta charset="UTF-8">
  <!-- 引入react核心代码 -->
  <script src="https://cdn.bootcss.com/react/16.8.6/umd/react.development.js"></script>
  <!-- 引入reactDom核心代码 -->
  <script src="https://cdn.bootcss.com/react-dom/16.8.6/umd/react-dom.development.js"></script>
  <!-- 引入babel核心代码 -->
  <script src="https://cdn.bootcss.com/babel-standalone/6.26.0/babel.min.js"></script>
  <title>JSX & React.createElement()</title>
</head>
<body>
  <!-- 使用javascript原生插入节点的根节点 -->
  <div id="rootByJs"></div>
  <!-- 使用React.createElement()方法插入节点的根节点 -->
  <div id="rootByReactCreateElement"></div>
  <!-- 使用JSX方法插入节点的根节点 -->
  <div id="rootByJsx"></div>
  <script>
    // 原生方法插入
    let htmlNode = document.createElement('div');
    htmlNode.innerHTML = '你好 前端战五渣';
    htmlNode.className = 'title';
    document.getElementById('rootByJs').appendChild(htmlNode);
  </script>
  <script>
    // 使用React.createElement()方法插入
    ReactDOM.render(
      React.createElement('div', {className: "title"}, '你好 前端战五渣'),
      document.getElementById('rootByReactCreateElement')
    );
  </script>
  <script type="text/babel">
    // 使用JSX方法插入
    ReactDOM.render(
      <div className="title">你好 前端战五渣</div>,
      document.getElementById('rootByJsx')
    );
  </script>
</body>
</html>
image

上面实现这个功能,用了三种方法,一种是js原生方法,一种是用react提供的createElement方法,还有最后一种使用JSX来实现的。

什么是JSX

其实jsx就是react这个框架提出的一种语法扩展,在react建议使用jsx,因为jsx可以清晰明了的描述DOM结构。可能到这里我们可能有人会说,这跟模板语言有什么区别呢?template也可以实现啊,但是JSX具有JavaScript的全部功能(官网这么说的🤦‍♀️)

<span style="font-weight: 600; color: red">一句话总结:JSX语法就是JavaScript和html可以混着写,灵活的一笔</span>

JSX的优点呢?

  1. 可以在js中写更加语义化且简单易懂的标签
  2. 更加简洁
  3. 结合原生js的语法

(也有人说jsx写起来很乱,仁者见仁智者见智吧)

JSX和React.createElement()的关系

那我们知道了JSX是什么,可是这跟我们这回要说的React.createElement()方法有什么关系呢?先来回顾一个面试会问的问题“你能说说vue和react有什么区别吗”,有一个区别就是在使用webpack打包的过程中,vue是用vue-loader来处理.vue后缀的文件,而react在打包的时候,是通过babel来转换的,因为react的组件说白了还是.js或者.jsx,是扩展的js语法,所以是通过babel转换成浏览器识别的es5或者其他版本的js

那我们来看看jsx的语法通过babel转换会变成什么样⬇️

image

我们可以看到通过babel转换以后,我们的JSX语法中的标签会被转换成一个React.createElement()并传入对应的参数

ReactDOM.render(
  <div className="title">hello gedesiwen</div>,
  document.getElementById('rootByJsx')
);

变~

ReactDOM.render(
  React.createElement('div', {className: 'title'}, 'hello gedesiwen'),
  document.getElementById('rootByJsx')
);

这我们看见了jsx变成了React.createElement()

多个子节点

上面的代码中,我们只是有一个子节点,就是文本节点“你好 前端战五渣”,那如果我们有很多个呢

我们在React组件中代码是这样的⬇️

import DragonBall from './dragonBall';

let htmlNode = (
  <Fragment>
    <DragonBall name="孙悟空"/>
    <div className="hello" key={1}>hello</div>
    <div className="world" key={2}>world</div>
  </Fragment>
)

ReactDOM.render(
  htmlNode,
  document.getElementById('rootByJsx')
);

我们的节点中包括DragonBall组件,还有Fragment,并且还有两个div

Fragment是干什么的呢???这就是JSX语法的一个规则,我们只能有一个根节点,如果我们有两个并列的div,但是直接写并列的两个div会报错,我们就只能在外面套一层div,但是我们不想创建不用的标签,这时候我们就能使用Fragment,他不会被渲染出来

React 中的一个常见模式是一个组件返回多个元素。Fragments 允许你将子列表分组,而无需向 DOM 添加额外节点。
————react文档

那上面这段我们通过babel会转换成这样⬇️

var htmlNode = React.createElement(
    Fragment,
    null,
    React.createElement(_dragonBall.default, {name: "saiyajin"}),
    React.createElement("div", {className: "hello", key: 1}, "hello"),
    React.createElement("div", {className: "world", key: 2}, "world")
);
ReactDOM.render(htmlNode, document.getElementById('rootByJsx'));

这就是我们转换完的js,那我们的React.createElement()方法到底做了什么呢

React.createElement()源码

首先我们需要从github上把react的源码,v16.8.6拉下来

然后我们找到在文件/packages/react/src/ReactElement.js这个文件中就有我们需要的React.createElement()方法

(代码中左右判断__DEV__的代码,不做考虑)

先上完整的方法代码,伴有注释

/**
 * React的创建元素方法
 * @param type 标签名字符串(如’div‘或'span'),也可以是React组件类型,或是React fragment类型
 * @param config 包含元素各个属性键值对的对象
 * @param children 包含元素的子节点或者子元素
 */
function createElement(type, config, children) {
  let propName; // 声明一个变量,储存后面循环需要用到的元素属性
  const props = {}; // 储存元素属性的键值对集合
  let key = null; // 储存元素的key值
  let ref = null; // 储存元素的ref属性
  let self = null;  // 下面文章介绍
  let source = null; // 下面文章介绍

  if (config != null) { // 判断config是否为空,看看是不是没有属性
    // hasValidRef()这个方法就是判断config有没有ref属性,有的话就赋值给之前定义好的ref变量
    if (hasValidRef(config)) {
      ref = config.ref;
    }
    // hasValidKey()这个方法就是判断config有没有key属性,有的话就赋值给之前定义好的key变量
    if (hasValidKey(config)) {
      key = '' + config.key; // key值看来还给转成了字符串😳
    }
    // __self和__source下面文章做介绍,实际也没搞明白是干嘛的
    self = config.__self === undefined ? null : config.__self;
    source = config.__source === undefined ? null : config.__source;
    // 现在就是要把config里面的属性都一个一个挪到props这个之前声明好的对象里面
    for (propName in config) {
      if (
        // 判断某个config的属性是不是原型上的
        hasOwnProperty.call(config, propName) &&  // 这行判断是不是原型链上的属性
        !RESERVED_PROPS.hasOwnProperty(propName) // 不能是原型链上的属性,也不能是key,ref,__self以及__source
      ) {
        props[propName] = config[propName]; // 乾坤大挪移,把config上的属性一个一个转到props里面
      }
    }
  }
  // 处理除了type和config属性剩下的其他参数
  const childrenLength = arguments.length - 2; // 抛去type和config,剩下的参数个数
  if (childrenLength === 1) { // 如果抛去type和config,就只剩下一个参数,就直接把这个参数的值赋给props.children
    props.children = children; // 一个参数的情况一般是只有一个文本节点
  } else if (childrenLength > 1) { // 如果不是一个呢??
    const childArray = Array(childrenLength); // 声明一个有剩下参数个数的数组
    for (let i = 0; i < childrenLength; i++) { // 然后遍历,把每个参数赋值到上面声明的数组里
      childArray[i] = arguments[i + 2];
    }
    props.children = childArray; // 最后把这个数组赋值给props.children
  } // 所以props.children要不是一个字符串,要不就是一个数组

  // 如果有type并且type有defaultProps属性就执行下面这段
  // 那defaultProps属性是啥呢??
  // 如果传进来的是一个组件,而不是div或者span这种标签,可能就会有props,从父组件传进来的值如果没有的默认值
  if (type && type.defaultProps) {
    const defaultProps = type.defaultProps;
    for (propName in defaultProps) { // 遍历,然后也放到props里面
      if (props[propName] === undefined) {
        props[propName] = defaultProps[propName];
      }
    }
  }
  // 所以props里面存的是config的属性值,然后还有children的属性,存的是字符串或者数组,还有一部分defaultProps的属性
  // 然后返回一个调用ReactElement执行方法,并传入刚才处理过的参数
  return ReactElement(
    type,
    key,
    ref,
    self,
    source,
    ReactCurrentOwner.current,
    props,
  );
}

React.createElement()方法的代码加注释就是上面这个,小伙伴们应该都能看懂了吧,只是其中其中还有__self__source以及type.defaultProps没有讲清楚,那我们下面会讲到,我们可以先来看看这个最后返回的ReactElement()方法

ReactElement()源码

这个方法很简单,就是添加一个判断为react元素类型的值,然后返回,

/**
 * @param {*} type
 * @param {*} props
 * @param {*} key
 * @param {string|object} ref
 * @param {*} owner
 * @param {*} self A *temporary* helper to detect places where `this` is
 * different from the `owner` when React.createElement is called, so that we
 * can warn. We want to get rid of owner and replace string `ref`s with arrow
 * functions, and as long as `this` and owner are the same, there will be no
 * change in behavior.
 * 
 * 这虽然说了用于判断this指向的,但是。。。。。方法里面也没有用到,不知道是干嘛的😳😳😳😳
 * 
 * @param {*} source An annotation object (added by a transpiler or otherwise)
 * indicating filename, line number, and/or other information.
 * 
 * 这个参数一样。。。。也没有用到啊。。。那我传进来是干嘛的,什么注释对象。。😳😳😳搞不懂
 * 
 * @internal
 */
const ReactElement = function(type, key, ref, self, source, owner, props) {
  const element = {
    $$typeof: REACT_ELEMENT_TYPE, // 声明一下是react的元素类型
    type: type,
    key: key,
    ref: ref,
    props: props,
    _owner: owner,
  };
  return element;
};

__self和__source

刚看到React.createElement()方法里面就用到了__self__source两个属性,当时还去查了一下react的文档

文档中也没有说是干嘛用的,然后查了一下issues

image

发现是这哥们提交的commit,好😳😳😳他说_self是用来判断this和owner是不是同一个指向巴拉巴拉的,他还说__source是什么注释对象,我也没看懂是干嘛的。。。。然后继续往下看,看到React.createElement()方法返回ReactElement()方法,并且把这些都传进去了。。。。

<span style="color: red;font-weight: bold;">ReactElement源码中竟然没有用这两个参数</span>

<span style="color: red;font-weight: bold;">大哥你开心就好🤡🤡🤡</span>

看到这篇文章的大佬有知道是干嘛的可以告诉我。。。。我反正现在是懵逼的😶😶😶

type.defaultProps

这个是什么呢,我们来看一段代码吧

import React, { Component } from 'react';
import ReactDom from 'react-dom';

class DragonBall extends Component {
  render() {
    return (
      <div>
        {this.props.name}
      </div>
    )
  }
}

ReactDom.render(<DragonBall />, document.getElementById('root'))

如果我这个DragonBall组件需要展示从props传过来,如果我们没传呢,就会是undefined,就什么都不显示,如果我们想设置默认值呢,可以这么写⬇️

import React, { Component } from 'react';
import ReactDom from 'react-dom';

class DragonBall extends Component {
  render() {
    return (
      <div>
        {this.props.name || '戈德斯文'}
      </div>
    )
  }
}

ReactDom.render(<DragonBall />, document.getElementById('root'))

就是像上面这样写,这样我们就进行了一次判断,如果props.name如果没有的话,就显示后面的“戈德斯文”,那还有没有什么别的办法呢??

想也知道啊,肯定就是我们说的这个defaultProps,这个东西怎么用呢⬇️

import React, { Component } from 'react';
import ReactDom from 'react-dom';

class DragonBall extends Component {
  render() {
    return (
      <div>
        {this.props.name}
      </div>
    )
  }
}

DragonBall.defaultProps = {
  name: '戈德斯文'
}

ReactDom.render(<DragonBall />, document.getElementById('root'))

我们只需要这样设置就可以,如果我们页面中很多地方需要用到props传进来的值,就不需要每个用到props值的地方都进行一次判断了

所以,在React.createElement()源码中

 if (type && type.defaultProps) {
    const defaultProps = type.defaultProps;
    for (propName in defaultProps) { // 遍历,然后也放到props里面
      if (props[propName] === undefined) {
        props[propName] = defaultProps[propName];
      }
    }
  }

这段代码就是把默认的props重新赋值。

回到开始

经过React.createElement()方法处理,并且经过ReactElement()方法洗礼,我们最开始的

let htmlNode = React.createElement(
  Fragment, 
  null, 
  React.createElement(_dragonBall.default, null), 
  React.createElement("div", null, "hello"), 
  React.createElement("div", null, "world")
);
ReactDOM.render(htmlNode, document.getElementById('rootByJsx'));

最后到底是变成什么样的呢?

{
    "key": null,
    "ref": null,
    "props": {
        "children": [{
            "key": null,
            "ref": null,
            "props": {
                "name": "saiyajin"
            },
            "_owner": null,
            "_store": {}
        }, {
            "type": "div",
            "key": "1",
            "ref": null,
            "props": {
                "className": "hello",
                "children": "hello"
            },
            "_owner": null,
            "_store": {}
        }, {
            "type": "div",
            "key": "2",
            "ref": null,
            "props": {
                "className": "world",
                "children": "world"
            },
            "_owner": null,
            "_store": {}
        }]
    },
    "_owner": null,
    "_store": {}
}

然后再经过ReactDom.render()方法渲染到页面上

ps:端午节快乐~~回家过节喽


我是前端战五渣,一个前端界的小学生。

参考

  1. 《剖析 React 源码:先热个身》
  2. React:issues adding __self and __source special props
  3. 《Understanding React Default Props》
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,039评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,223评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,916评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,009评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,030评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,011评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,934评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,754评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,202评论 1 309
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,433评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,590评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,321评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,917评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,568评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,738评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,583评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,482评论 2 352