同作为MVVM框架,React相比于Vue来讲,上手更需要JavaScript 功底深厚一些,本系列将阅读React 相关源码,从jsx -> VDom -> RDOM 等一些列的过程,将会在本系列中一一讲解
工欲善其事,必先利其器
经过多年的发展,React已经更新了大版本16、17、18 ,本系列主要讲的是 version:17.0.2 ,在讲这个版本之前,我们先看一看在babel 的编译下,每个大版本之下会有什么样的变化。
jsx
<div className='box'>
<h1 className='title' style={{'color':'red'}}>React源码解析</h1>
<ul>
<li>第一章</li>
<li>第二章</li>
<li>第三章</li>
<li>第四章</li>
</ul>
</div>
v16.x及以前版本
v17及之后版本
所以各位看到了,在v16 及以前我们babel 进行jsx 解析编译的是根据@babel/babel-preset-react-app解析成React.createElement 进行包裹的,而v17 以及之后的版本,官网早就说明,对jsx 的转换用react/jsx-runtime ,而不再依赖React.createElement 了,看到这里我想各位对不同版本的babel解析jsx已经有了眉目了,早已经迫不及待想去看看jsx-runtime和createElement到底是如何玩的,那么进入源码
在babel解析后的v17产物中我们可以看得到 var _jsxRuntime = require("react/jsx-runtime"); 那么我们追本溯源可以找到在packages/react/src/jsx/ReactJSX.js 里面的jsxs 是怎么来的
import {REACT_FRAGMENT_TYPE} from 'shared/ReactSymbols';
import {
jsxWithValidationStatic,
jsxWithValidationDynamic,
jsxWithValidation,
} from './ReactJSXElementValidator';
import {jsx as jsxProd} from './ReactJSXElement';
const jsx = __DEV__ ? jsxWithValidationDynamic : jsxProd;
const jsxs = __DEV__ ? jsxWithValidationStatic : jsxProd;
const jsxDEV = __DEV__ ? jsxWithValidation : undefined;
export {REACT_FRAGMENT_TYPE as Fragment, jsx, jsxs, jsxDEV};
在非dev 环境下我们继续去找jsProd
export function jsx(type, config, maybeKey) {
let propName;
const props = {};
let key = null;
let ref = null;
if (maybeKey !== undefined) {
key = '' + maybeKey;
}
if (hasValidKey(config)) {
key = '' + config.key;
}
if (hasValidRef(config)) {
ref = config.ref;
}
for (propName in config) {
if (
hasOwnProperty.call(config, propName) &&
!RESERVED_PROPS.hasOwnProperty(propName)
) {
props[propName] = config[propName];
}
}
if (type && type.defaultProps) {
const defaultProps = type.defaultProps;
for (propName in defaultProps) {
if (props[propName] === undefined) {
props[propName] = defaultProps[propName];
}
}
}
return ReactElement(
type,
key,
ref,
undefined,
undefined,
ReactCurrentOwner.current,
props
)
}
ReactElement
const ReactElement = function(type, key, ref, self, source, owner, props) {
const element = {
$$typeof: REACT_ELEMENT_TYPE,
type: type,
key: key,
ref: ref,
props: props,
_owner: owner,
};
if (__DEV__) {
element._store = {};
Object.defineProperty(element._store, 'validated', {
configurable: false,
enumerable: false,
writable: true,
value: false,
});
Object.defineProperty(element, '_self', {
configurable: false,
enumerable: false,
writable: false,
value: self,
});
Object.defineProperty(element, '_source', {
configurable: false,
enumerable: false,
writable: false,
value: source,
});
if (Object.freeze) {
Object.freeze(element.props);
Object.freeze(element);
}
}
return element;
};
这上面便是v17 及之后版本的jsx-runtime 所做的事情。那么这里再去看一下v16 中的createElement 所做的事情吧。
相关参考视频讲解:进入学习
React.createElement
export function createElement(type, config, children) {
let propName;
const props = {};
let key = null;
let ref = null;
let self = null;
let source = null;
if (config != null) {
if (hasValidRef(config)) {
ref = config.ref;
if (__DEV__) {
warnIfStringRefCannotBeAutoConverted(config);
}
}
if (hasValidKey(config)) {
key = '' + config.key;
}
self = config.__self === undefined ? null : config.__self;
source = config.__source === undefined ? null : config.__source;
for (propName in config) {
if (
hasOwnProperty.call(config, propName) &&
!RESERVED_PROPS.hasOwnProperty(propName)
) {
props[propName] = config[propName];
}
}
}
const childrenLength = arguments.length - 2;
if (childrenLength === 1) {
props.children = children;
} else if (childrenLength > 1) {
const childArray = Array(childrenLength);
for (let i = 0; i < childrenLength; i++) {
childArray[i] = arguments[i + 2];
}
if (__DEV__) {
if (Object.freeze) {
Object.freeze(childArray);
}
}
props.children = childArray;
}
if (type && type.defaultProps) {
const defaultProps = type.defaultProps;
for (propName in defaultProps) {
if (props[propName] === undefined) {
props[propName] = defaultProps[propName];
}
}
}
if (__DEV__) {
if (key || ref) {
const displayName =
typeof type === 'function'
? type.displayName || type.name || 'Unknown'
: type;
if (key) {
defineKeyPropWarningGetter(props, displayName);
}
if (ref) {
defineRefPropWarningGetter(props, displayName);
}
}
}
return ReactElement(
type,
key,
ref,
self,
source,
ReactCurrentOwner.current,
props,
);
}
由React.createElement源码得知,他做了如下事情
-
解析config 参数中是否有合法的 key 、ref 属性,并处理,并将其他的属性挂到props 上。 -
解析函数的第三参数,并分情况将第三参数挂到props.children 上。 -
对默认props进行处理,如果存在该属性则直接挂载到props上,不存在则要添加上。 -
开发环境下将 _store、_self、_source 设置为不可枚举状态,为后期的diff比较作优化,提高比较性能。 -
将type、key、ref、props 等属性通过调用ReactElement函数创建虚拟dom。
ReactElement
const ReactElement = function(type, key, ref, self, source, owner, props) {
const element = {
$$typeof: REACT_ELEMENT_TYPE,
type: type,
key: key,
ref: ref,
props: props,
_owner: owner,
};
if (__DEV__) {
element._store = {};
Object.defineProperty(element._store, 'validated', {
configurable: false,
enumerable: false,
writable: true,
value: false,
});
Object.defineProperty(element, '_self', {
configurable: false,
enumerable: false,
writable: false,
value: self,
});
Object.defineProperty(element, '_source', {
configurable: false,
enumerable: false,
writable: false,
value: source,
});
if (Object.freeze) {
Object.freeze(element.props);
Object.freeze(element);
}
}
return element;
};
仔细瞧一瞧,这个其实跟jsxs 调用的ReactElement 实现的差不多的功能,但是为什么要写两遍?仔细看来,在两个版本的ReactElement 中,传入的参数不一致,在开发环境下,分别对其做劫持不可枚举状态,仅此而已 ?
React.Component
写惯了hooks 组件,但是Class 组件也别忘了哟,因为在React17 里面Class 组件也是没有被抹去的,所以既然是源码解析,那么我们也要来看一看这个Component 到底干了啥。
function Component(props, context, updater) {
this.props = props;
this.context = context;
this.refs = emptyObject;
this.updater = updater || ReactNoopUpdateQueue;
}
Component.prototype.isReactComponent = {};
Component.prototype.setState = function(partialState, callback) {
invariant(
typeof partialState === 'object' ||
typeof partialState === 'function' ||
partialState == null,
'setState(...): takes an object of state variables to update or a ' +
'function which returns an object of state variables.',
);
this.updater.enqueueSetState(this, partialState, callback, 'setState');
};
Component.prototype.forceUpdate = function(callback) {
this.updater.enqueueForceUpdate(this, callback, 'forceUpdate');
};
从源码上可以得知,React.Component 主要做了以下几件事情:
- 将
props, context, updater 挂载到this 上,props,context 一目了然,后面的updater 位触发器,上面挂了很多方法,我们后面再谈。 - 在
Component 原型链上添加 isReactComponent 对象,用于区分函数组件还是类组件。 - 在
Component 原型链上添加 setState 方法,触发更新。 - 在
Component 原型链上添加 forceUpdate 方法,强制更新。
总结
不管是类组件还是函数组件,最终我们写的jsx 都被babel 转化成了可识别的元素,其中我们也看了ReactElement ,createElement ,Component 等内部实现,了解到了作为ReactElement 他是怎么被创建的,但是远远没有完,因为我们知道我们在写React的时候,会在后面带上一个ReactDOM.render(<Element/>, 'root') ,没错我们下一章节就要去探索一下ReactDOM.render 方法了。
|