摘要:這個(gè)函數(shù)內(nèi)處理了的生命周期以及和生命周期鉤子函數(shù),調(diào)用返回實(shí)際要渲染的內(nèi)容,如果內(nèi)容是復(fù)合組件,仍然會(huì)調(diào)用,復(fù)合組件最終一定會(huì)返回原生組件,并且最終調(diào)用的函數(shù)生成要渲染的。
原文鏈接地址:https://github.com/Nealyang%EF%BC%9A%E7%BB%84%E4%BB%B6%E7%9A%84%E5%88%9D%E5%A7%8B%E5%8C%96%E4%B8%8E%E6%B8%B2%E6%9F%93.md) 轉(zhuǎn)載請(qǐng)注明出處前言
戰(zhàn)戰(zhàn)兢兢寫下開篇...也感謝小蘑菇大神以及網(wǎng)上各路大神的博客資料參考~
閱讀源碼的方式有很多種,廣度優(yōu)先法、調(diào)用棧調(diào)試法等等,此系列文章,采用基線法,顧名思義,就是以低版本為基線,逐漸了解源碼的演進(jìn)過(guò)程和思路。
react最初的設(shè)計(jì)靈感來(lái)源于游戲渲染的機(jī)制:當(dāng)數(shù)據(jù)變化時(shí),界面僅僅更新變化的部分而形成新的一幀渲染。所以設(shè)計(jì)react的核心就是認(rèn)為UI只是把數(shù)據(jù)通過(guò)映射關(guān)系變換成另一種形式的數(shù)據(jù),也就是展示方式。傳統(tǒng)上,web架構(gòu)使用模板或者HTML指令構(gòu)造頁(yè)面。react則處理構(gòu)建用戶界面通過(guò)將他們份極為virtual dom,當(dāng)然這也是react的核心,整個(gè)react架構(gòu)的設(shè)計(jì)理念也是為此展開的。
準(zhǔn)備工作我們采用基線法去學(xué)習(xí)react源碼,所以目前基于的版本為stable-0.3,后面我們?cè)谥鸩椒治鰧W(xué)習(xí)演變的版本。
clone代碼git clone https://github.com/facebook/react.git git checkout 0.3-stable
React源代碼都在src目錄中,src包含了8個(gè)目錄,其主要內(nèi)容描述見(jiàn)下表。
目 錄 | 內(nèi)容 |
---|---|
core | React 核心類 |
domUtil | Dom操作和CSS操作的相關(guān)工具類 |
environment | 當(dāng)前JS執(zhí)行環(huán)境的基本信息 |
event | React事件機(jī)制的核心類 |
eventPlugins | React事件機(jī)制的事件綁定插件類 |
test | 測(cè)試目錄 |
utils | 各種工具類 |
vendor | 可替換模塊存放目錄 |
我們將該版本編譯后的代碼放到example下,引入到basic/index.html中運(yùn)行調(diào)試。
組件初始化 使用這里還是以basic.html中的代碼為例
回到我們說(shuō)的組件初始化,抽離下上面的代碼就是:
var ExampleApplication = React.createClass({render:function(){ returnNealyang}})
熟悉react使用的人都知道,render方法不能為空,當(dāng)然,createClass中我們也可以去寫一寫生命周期的鉤子函數(shù),這里我們暫且省略,畢竟目前我們更加的關(guān)注react組建的初始化過(guò)程。
同樣,熟悉react使用方法的人也會(huì)有疑惑了,怎么實(shí)例代碼中的render最后return的是React.DOM.p(null,message)
所以到這里,就不得不說(shuō)一下react的編譯階段了
編譯階段我們都知道,在js中直接編寫html代碼,或者。。。jsx語(yǔ)法這樣的AST,在js詞法分析階段就會(huì)拋出異常的。
對(duì)的,所以我們?cè)诰帉憆eact代碼的時(shí)候都會(huì)借助babel去轉(zhuǎn)碼
從babel官網(wǎng)上寫個(gè)例子即可看出:
對(duì)呀!明明人家用的是react.createElement方法,我們?cè)趺闯霈F(xiàn)個(gè)React.DOM.p...
OK,歷史原因:
react現(xiàn)在版本中,使用babel-preset-react來(lái)編譯jsx,這個(gè)preset又包含了4個(gè)插件,其中transform-react-jsx負(fù)責(zé)編譯jsx,調(diào)用了React.createElement函數(shù)生成虛擬組件
在react-0.3里,編譯結(jié)果稍稍有些不同,官方給出的示例文件,使用JSXTransformer.js編譯jsx(也就是),對(duì)于native組件和composite組件編譯的方式也不一致。也就是我們看到的React.DOM.p or ReactComponsiteComponent
native組件:編譯成React.DOM.xxx(xxx如div),函數(shù)運(yùn)行返回一個(gè)ReactNativeComponent實(shí)例。
composite組件:編譯成createClass返回的函數(shù)調(diào)用,函數(shù)運(yùn)行返回一個(gè)ReactCompositeComponent實(shí)例
題外話,不管用什么框架,到瀏覽器這部分的,什么花里胡哨的都不復(fù)存在。我這就是js、css、html。所以我們這里的ReactCompositeComponent最終其實(shí)還是需要轉(zhuǎn)成原生元素的 。
組件創(chuàng)建從React.js中我們可以找到createClass的出處:
"use strict"; var ReactCompositeComponent = require("ReactCompositeComponent"); ... var React = { ... createClass: ReactCompositeComponent.createClass, ... }; module.exports = React;
createClass 代碼
var ReactCompositeComponentBase = function() {}; function mixSpecIntoComponent(Constructor, spec) { var proto = Constructor.prototype; for (var name in spec) { if (!spec.hasOwnProperty(name)) { continue; } var property = spec[name]; var specPolicy = ReactCompositeComponentInterface[name]; if (RESERVED_SPEC_KEYS.hasOwnProperty(name)) { RESERVED_SPEC_KEYS[name](Constructor, property); } else if (property && property.__reactAutoBind) { if (!proto.__reactAutoBindMap) { proto.__reactAutoBindMap = {}; } proto.__reactAutoBindMap[name] = property.__reactAutoBind; } else if (proto.hasOwnProperty(name)) { // For methods which are defined more than once, call the existing methods // before calling the new property. proto[name] = createChainedFunction(proto[name], property); } else { proto[name] = property; } } } createClass: function (spec) { var Constructor = function (initialProps, children) { this.construct(initialProps, children); }; // ReactCompositeComponentBase是React復(fù)合組件的原型函數(shù) Constructor.prototype = new ReactCompositeComponentBase(); Constructor.prototype.constructor = Constructor; // 把消費(fèi)者聲明配置spec合并到Constructor.prototype中 mixSpecIntoComponent(Constructor, spec); // 判斷合并后的結(jié)果有沒(méi)有render,如果沒(méi)有 render,拋出一個(gè)異常 invariant( Constructor.prototype.render, "createClass(...): Class specification must implement a `render` method." ); //工廠 var ConvenienceConstructor = function (props, children) { return new Constructor(props, children); }; ConvenienceConstructor.componentConstructor = Constructor; ConvenienceConstructor.originalSpec = spec; return ConvenienceConstructor; },
mixSpecIntoComponent 方法就是講spec的屬性賦值給Constructor的原型上
createClass返回一個(gè)ConvenienceConstructor構(gòu)造函數(shù),構(gòu)造函數(shù)接受props、children 構(gòu)造函數(shù)的靜態(tài)方法componentConstructor和originalSpec分別指向Constructor和spec。
有種類似于寄生組合式繼承的寫法,Constructor為每一個(gè)組件實(shí)例的原型(`var instance = new Constructor();
instance.construct.apply(instance, arguments);`)。Constructor原型指向ReactCompositeComponentBase,又把構(gòu)造器指向Constructor自己。然后把傳入的spec合并到Constructor.prototype中。判斷合并后的結(jié)果有沒(méi)有render,如果沒(méi)有 render,拋出一個(gè)異常
其實(shí)很多人看到這估計(jì)都會(huì)很疑惑,為毛這樣搞???直接返回個(gè)構(gòu)造函數(shù)不就可以了嘛。
其實(shí)react在后面做diff算法的時(shí)候,是采用組件的Constructor來(lái)判斷組件是否相同的。如此可以保證每個(gè)createClass創(chuàng)建出來(lái)的組件都是一個(gè)新的Constructor。
ok,那么我直接用寄生繼承呀
// 寫法1 const createClass = function(spec) { var Constructor = function (initialProps, children) { this.construct(initialProps, children); }; Constructor.prototype = new ReactCompositeComponentBase(); Constructor.prototype.constructor = Constructor; mixSpecIntoComponent(ReactCompositeComponentBase, spec) return Constructor } const Table1 = new createClass(spec)(props, children); //console.log(Table1.constructor)
為什么還需要ConvenienceConstructor呢?說(shuō)實(shí)話,我也不知道,然后看了在網(wǎng)上查到相關(guān)信息說(shuō)道:
上面寫法在大多數(shù)情況下并不會(huì)產(chǎn)生什么問(wèn)題,但是,當(dāng)團(tuán)隊(duì)里的人無(wú)意中修改錯(cuò)點(diǎn)什么,比如:
Table1.prototype.onClick = null
這樣,所有Table1實(shí)例化的組件,onClick全部為修改后的空值
我們知道,js是動(dòng)態(tài)解釋型語(yǔ)言,函數(shù)可以運(yùn)行時(shí)被隨意篡改。而靜態(tài)編譯語(yǔ)言在運(yùn)行時(shí)期間,函數(shù)不可修改(某些靜態(tài)語(yǔ)言也可以修改)。所以采用這種方式防御用戶對(duì)代碼的篡改。
組件實(shí)例化既然createClass返回的是一個(gè)構(gòu)造函數(shù),那么我們就來(lái)看看他的實(shí)例化吧
/** * Base constructor for all React component. * * Subclasses that override this method should make sure to invoke * `ReactComponent.Mixin.construct.call(this, ...)`. * * @param {?object} initialProps * @param {*} children * @internal */ construct: function (initialProps, children) { this.props = initialProps || {}; if (typeof children !== "undefined") { this.props.children = children; } // Record the component responsible for creating this component. this.props[OWNER] = ReactCurrentOwner.current; // All components start unmounted. this._lifeCycleState = ComponentLifeCycle.UNMOUNTED; },
其實(shí)也就是將props、children掛載到this.props上 以及生命周期的設(shè)置。這里暫且不說(shuō),因?yàn)槲乙舱诳?。。。哇咔?/p>
這里的
this.props[OWNER] = ReactCurrentOwner.current;
this.props[OWNER]指的是當(dāng)前組件的容器(父)組件實(shí)例
如果我們直接在basic.html中打印就直接出來(lái)的是null,但是如果像如下的方式書寫:
const Children = React.createClass({ componentDidMount = () => console.log(this.props["{owner}"]), render = () => null }) const Parent = React.createClass({ render: () =>})
這里輸出的就是Parent組件實(shí)例。
再看看ReactCurrentOwner.current的賦值就明白了
_renderValidatedComponent: function () { ReactCurrentOwner.current = this; var renderedComponent = this.render(); ReactCurrentOwner.current = null; invariant( ReactComponent.isValidComponent(renderedComponent), "%s.render(): A valid ReactComponent must be returned.", this.constructor.displayName || "ReactCompositeComponent" ); return renderedComponent; }
可以看出來(lái),在執(zhí)行render前后,分別設(shè)置了ReactCurrentOwner.current的值,這樣就能保證render函數(shù)內(nèi)的子組件能賦上當(dāng)前組件的實(shí)例,也就是this。
組件渲染我們先撇開事務(wù)、事件池、生命周期、diff當(dāng)然也包括fiber 等,先不談,其實(shí)渲染就是將經(jīng)過(guò)babel編譯后的,當(dāng)然這里是JSXTransformer.js編譯后的Ojb給寫入到HTML中而已。
export default function render(vnode, parent) { let dom; if (typeof vnode === "string") { dom = document.createTextNode(vnode); // let span_dom = document.createElement("span") // span_dom.appendChild(dom); // parent.appendChild(span_dom); parent.appendChild(dom); } else if (typeof vnode.nodeName === "string") { dom = document.createElement(vnode.nodeName); setAttrs(dom, vnode.props); parent.appendChild(dom) for(let i = 0; i < vnode.children.length; i++) { render(vnode.children[i], dom) } }else if(typeof vnode.nodeName === "function"){ let innerVnode = vnode.nodeName.prototype.render(); render(innerVnode,parent) } } function setAttrs(dom, props) { const ALL_KEYS = Object.keys(props); ALL_KEYS.forEach(k =>{ const v = props[k]; // className if(k === "className"){ dom.setAttribute("class",v); return; } if(k == "style") { if(typeof v == "string") { dom.style.cssText = v } if(typeof v == "object") { for (let i in v) { dom.style[i] = v[i] } } return } if(k[0] == "o" && k[1] == "n") { const capture = (k.indexOf("Capture") != -1) dom.addEventListener(k.substring(2).toLowerCase(),v,capture) return } dom.setAttribute(k, v) }) }
是的,就這樣
OK,回到源碼~
在我們目前使用的react版本中,渲染調(diào)用的是ReactDOM.render方法,這里ReactMount.renderComponent為我們的入口方法。
ReactMount.renderComponent在react初探章節(jié)講過(guò)。如果組件渲染過(guò),就更新組件屬性,如果組件沒(méi)有渲染過(guò),掛載組件事件,并把虛擬組件渲染成真實(shí)組件插入container內(nèi)。通常,我們很少去調(diào)用兩次renderComponent,所以大多數(shù)情況下不會(huì)更新組件屬性而是新創(chuàng)建dom節(jié)點(diǎn)并插入到container中。
ReactComponent.mountComponentIntoNode之內(nèi)開啟了一個(gè)事務(wù),事務(wù)保證渲染階段不會(huì)有任何事件觸發(fā),并阻斷的componentDidMount事件,待執(zhí)行后執(zhí)行等,事務(wù)在功能一章我們會(huì)詳細(xì)講解,這里不細(xì)討論。
ReactComponent._mountComponentIntoNode這個(gè)函數(shù)調(diào)用mountComponent獲得要渲染的innerHTML,然后更新container的innerHTML。
ReactCompositeComponent.mountComponent是最主要的邏輯方法。這個(gè)函數(shù)內(nèi)處理了react的生命周期以及componentWillComponent和componentDidMount生命周期鉤子函數(shù),調(diào)用render返回實(shí)際要渲染的內(nèi)容,如果內(nèi)容是復(fù)合組件,仍然會(huì)調(diào)用mountComponent,復(fù)合組件最終一定會(huì)返回原生組件, 并且最終調(diào)用ReactNativeComponent的mountComponent函數(shù)生成要渲染的innerHTML。
renderComponent: function(nextComponent, container) { var prevComponent = instanceByReactRootID[getReactRootID(container)]; if (prevComponent) { var nextProps = nextComponent.props; ReactMount.scrollMonitor(container, function() { prevComponent.replaceProps(nextProps); }); return prevComponent; } ReactMount.prepareTopLevelEvents(ReactEventTopLevelCallback); var reactRootID = ReactMount.registerContainer(container); instanceByReactRootID[reactRootID] = nextComponent; nextComponent.mountComponentIntoNode(reactRootID, container); return nextComponent; },
這段代碼邏輯大概就是上面的流程圖,這里不再贅述。
mountComponentIntoNode
從debugger中,可以看出mountComponentIntoNode第一個(gè)參數(shù)其實(shí)傳入的是react分配給組件的一個(gè)唯一標(biāo)識(shí)
mountComponentIntoNode: function(rootID, container) { var transaction = ReactComponent.ReactReconcileTransaction.getPooled(); transaction.perform( this._mountComponentIntoNode, this, rootID, container, transaction ); ReactComponent.ReactReconcileTransaction.release(transaction); },
源碼中,這里跟事務(wù)扯到了關(guān)系,其實(shí)我們只要關(guān)注渲染本身,所以這里我們直接看this._mountComponentIntoNode的方法實(shí)現(xiàn)
_mountComponentIntoNode
_mountComponentIntoNode: function(rootID, container, transaction) { var renderStart = Date.now(); var markup = this.mountComponent(rootID, transaction); ReactMount.totalInstantiationTime += (Date.now() - renderStart); var injectionStart = Date.now(); // Asynchronously inject markup by ensuring that the container is not in // the document when settings its `innerHTML`. var parent = container.parentNode; if (parent) { var next = container.nextSibling; parent.removeChild(container); container.innerHTML = markup; if (next) { parent.insertBefore(container, next); } else { parent.appendChild(container); } } else { container.innerHTML = markup; } ReactMount.totalInjectionTime += (Date.now() - injectionStart); },
上述代碼流程大概如下:
流程的確如上,作為一個(gè)初探源碼者,我當(dāng)然不關(guān)心你到底是在哪innerHTML的,我想知道你是腫么把jsx編譯后的Obj轉(zhuǎn)成HTML的哇~
ReactCompositeComponent.mountComponent
這里類變成了ReactCompositeComponent(debugger可以跟蹤每一個(gè)函數(shù))
源碼中的this.mountComponent,為什么不是調(diào)用ReactComponent.mountComponent呢?這里主要使用了多重繼承機(jī)制(Mixin,后續(xù)講解)。
mountComponent: function(rootID, transaction) { // 掛在組件ref(等于當(dāng)前組件實(shí)例)到this.refs上 ReactComponent.Mixin.mountComponent.call(this, rootID, transaction); // Unset `this._lifeCycleState` until after this method is finished. // 這是生命周期 this._lifeCycleState = ReactComponent.LifeCycle.UNMOUNTED; this._compositeLifeCycleState = CompositeLifeCycle.MOUNTING; // 組件聲明有props,執(zhí)行校驗(yàn) if (this.constructor.propDeclarations) { this._assertValidProps(this.props); } // 為組件聲明時(shí)間綁定this if (this.__reactAutoBindMap) { this._bindAutoBindMethods(); } //獲取state this.state = this.getInitialState ? this.getInitialState() : null; this._pendingState = null; // 如果組件聲明componentWillMount函數(shù),執(zhí)行并把setState的結(jié)果更新到this.state上 if (this.componentWillMount) { this.componentWillMount(); // When mounting, calls to `setState` by `componentWillMount` will set // `this._pendingState` without triggering a re-render. if (this._pendingState) { this.state = this._pendingState; this._pendingState = null; } } // 如果聲明了componentDidMount,則把其加入到ReactOnDOMReady隊(duì)列中 if (this.componentDidMount) { transaction.getReactOnDOMReady().enqueue(this, this.componentDidMount); } // 調(diào)用組件聲明的render函數(shù),并返回ReactComponent抽象類實(shí)例(ReactComponsiteComponent或 // ReactNativeComponent),調(diào)用相應(yīng)的mountComponent函數(shù) this._renderedComponent = this._renderValidatedComponent(); // Done with mounting, `setState` will now trigger UI changes. this._compositeLifeCycleState = null; this._lifeCycleState = ReactComponent.LifeCycle.MOUNTED; return this._renderedComponent.mountComponent(rootID, transaction); },
這個(gè)函數(shù)式VDom中最為重要的函數(shù),操作也最為復(fù)雜,執(zhí)行操作大概如下:
如上,很多內(nèi)容跟我們這part有點(diǎn)超綱。當(dāng)然,后面都會(huì)說(shuō)道,關(guān)于react的渲染,其實(shí)我們的工作很簡(jiǎn)單,不關(guān)于任何,在拿到render的東西后,如何解析,其實(shí)就是最后一行代碼:this._renderedComponent.mountComponent(rootID, transaction);
mountComponent: function(rootID, transaction) { ReactComponent.Mixin.mountComponent.call(this, rootID, transaction); assertValidProps(this.props); return ( this._createOpenTagMarkup() + this._createContentMarkup(transaction) + this._tagClose ); }, _createOpenTagMarkup: function() { var props = this.props; var ret = this._tagOpen; for (var propKey in props) { if (!props.hasOwnProperty(propKey)) { continue; } var propValue = props[propKey]; if (propValue == null) { continue; } if (registrationNames[propKey]) { putListener(this._rootNodeID, propKey, propValue); } else { if (propKey === STYLE) { if (propValue) { propValue = props.style = merge(props.style); } propValue = CSSPropertyOperations.createMarkupForStyles(propValue); } var markup = DOMPropertyOperations.createMarkupForProperty(propKey, propValue); if (markup) { ret += " " + markup; } } } return ret + " id="" + this._rootNodeID + "">"; }, /** * Creates markup for the content between the tags. * * @private * @param {ReactReconcileTransaction} transaction * @return {string} Content markup. */ _createContentMarkup: function(transaction) { // Intentional use of != to avoid catching zero/false. var innerHTML = this.props.dangerouslySetInnerHTML; if (innerHTML != null) { if (innerHTML.__html != null) { return innerHTML.__html; } } else { var contentToUse = this.props.content != null ? this.props.content : CONTENT_TYPES[typeof this.props.children] ? this.props.children : null; var childrenToUse = contentToUse != null ? null : this.props.children; if (contentToUse != null) { return escapeTextForBrowser(contentToUse); } else if (childrenToUse != null) { return this.mountMultiChild( flattenChildren(childrenToUse), transaction ); } } return ""; }, function ReactNativeComponent(tag, omitClose) { this._tagOpen = "<" + tag + " "; this._tagClose = omitClose ? "" : "" + tag + ">"; this.tagName = tag.toUpperCase(); }
代碼稍微多一點(diǎn),但是工作目標(biāo)很單一,就是為了將描述jsx的obj解析成HTML string。其實(shí)可以參照我上面直接亮出來(lái)的自己寫的代碼部分。
如上,其實(shí)我們已經(jīng)完成了組件的初始化、渲染~
好吧,我們一直說(shuō)的渲染的核心部分還沒(méi)有細(xì)說(shuō)~~~
掛載組件ref到this.refs上,設(shè)置生命周期、狀態(tài)和rootIDmountComponent: function(rootID, transaction) { invariant( this._lifeCycleState === ComponentLifeCycle.UNMOUNTED, "mountComponent(%s, ...): Can only mount an unmounted component.", rootID ); var props = this.props; if (props.ref != null) { ReactOwner.addComponentAsRefTo(this, props.ref, props[OWNER]); } this._rootNodeID = rootID; this._lifeCycleState = ComponentLifeCycle.MOUNTED; // Effectively: return ""; },
如果組件ref屬性為空,則為組件的this.refs上掛在當(dāng)前組件,也就是this,實(shí)現(xiàn)如下:
addComponentAsRefTo: function(component, ref, owner) { owner.attachRef(ref, component); }
attachRef: function(ref, component) { var refs = this.refs || (this.refs = {}); refs[ref] = component; },
上述代碼我刪除了相關(guān)的判斷警告。
設(shè)置組件生命狀態(tài)組件的生命狀態(tài)和生命周期鉤子函數(shù)是react的兩個(gè)概念,在react中存在兩種生命周期
主:組件生命周期:_lifeCycleState,用來(lái)校驗(yàn)react組件在執(zhí)行函數(shù)時(shí)狀態(tài)值是否正確
輔:復(fù)合組件生命周期:_componsiteLifeCycleState,用來(lái)保證setState流程不受其他行為影響
_lifeCycleStatevar ComponentLifeCycle = keyMirror({ /** * Mounted components have a DOM node representation and are capable of * receiving new props. */ MOUNTED: null, /** * Unmounted components are inactive and cannot receive new props. */ UNMOUNTED: null });
組件生命周期非常簡(jiǎn)單,就枚舉了兩種,MOUNTED and UNMOUNTED
在源碼中使用其只是為了在相應(yīng)的階段觸發(fā)時(shí)候校驗(yàn),并且給出錯(cuò)誤提示
getDOMNode: function() { invariant( ExecutionEnvironment.canUseDOM, "getDOMNode(): The DOM is not supported in the current environment." ); invariant( this._lifeCycleState === ComponentLifeCycle.MOUNTED, "getDOMNode(): A component must be mounted to have a DOM node." ); var rootNode = this._rootNode; if (!rootNode) { rootNode = document.getElementById(this._rootNodeID); if (!rootNode) { // TODO: Log the frequency that we reach this path. rootNode = ReactMount.findReactRenderedDOMNodeSlow(this._rootNodeID); } this._rootNode = rootNode; } return rootNode; },_compositeLifeCycleState
復(fù)合組件的生命周期只在一個(gè)地方使用:setState
var CompositeLifeCycle = keyMirror({ /** * Components in the process of being mounted respond to state changes * differently. */ MOUNTING: null, /** * Components in the process of being unmounted are guarded against state * changes. */ UNMOUNTING: null, /** * Components that are mounted and receiving new props respond to state * changes differently. */ RECEIVING_PROPS: null, /** * Components that are mounted and receiving new state are guarded against * additional state changes. */ RECEIVING_STATE: null });
replaceState: function(completeState) { var compositeLifeCycleState = this._compositeLifeCycleState; invariant( this._lifeCycleState === ReactComponent.LifeCycle.MOUNTED || compositeLifeCycleState === CompositeLifeCycle.MOUNTING, "replaceState(...): Can only update a mounted (or mounting) component." ); invariant( compositeLifeCycleState !== CompositeLifeCycle.RECEIVING_STATE && compositeLifeCycleState !== CompositeLifeCycle.UNMOUNTING, "replaceState(...): Cannot update while unmounting component or during " + "an existing state transition (such as within `render`)." ); this._pendingState = completeState; // Do not trigger a state transition if we are in the middle of mounting or // receiving props because both of those will already be doing this. if (compositeLifeCycleState !== CompositeLifeCycle.MOUNTING && compositeLifeCycleState !== CompositeLifeCycle.RECEIVING_PROPS) { this._compositeLifeCycleState = CompositeLifeCycle.RECEIVING_STATE; var nextState = this._pendingState; this._pendingState = null; var transaction = ReactComponent.ReactReconcileTransaction.getPooled(); transaction.perform( this._receivePropsAndState, this, this.props, nextState, transaction ); ReactComponent.ReactReconcileTransaction.release(transaction); this._compositeLifeCycleState = null; } },
setState會(huì)調(diào)用replaceState ,然后調(diào)用_receivePropsAndState來(lái)更新界面
如果組件正處在mounting的過(guò)程或者接受props的過(guò)程中,那么將state緩存在_pendingState中,并不會(huì)更新界面的值。
校驗(yàn)props_assertValidProps: function(props) { var propDeclarations = this.constructor.propDeclarations; var componentName = this.constructor.displayName; for (var propName in propDeclarations) { var checkProp = propDeclarations[propName]; if (checkProp) { checkProp(props, propName, componentName); } } }
this.constructor.propDeclarations 就是組件聲明的props屬性,由于props是運(yùn)行時(shí)傳入的屬性。我們可以看到聲明props的屬性值即為checkProp
結(jié)束語(yǔ)其實(shí)至此,關(guān)于本篇組件的初始化、渲染已經(jīng)介紹完畢,由于代碼中關(guān)于太多后續(xù)章節(jié),生命周期、props、state、對(duì)象緩沖池、事務(wù)等,所以暫時(shí)都先略過(guò),后續(xù)學(xué)習(xí)到的時(shí)候再回頭查閱。
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/97093.html
摘要:異步實(shí)戰(zhàn)狀態(tài)管理與組件通信組件通信其他狀態(tài)管理當(dāng)需要改變應(yīng)用的狀態(tài)或有需要更新時(shí),你需要觸發(fā)一個(gè)把和載荷封裝成一個(gè)。的行為是同步的。所有的狀態(tài)變化必須通過(guò)通道。前端路由實(shí)現(xiàn)與源碼分析設(shè)計(jì)思想應(yīng)用是一個(gè)狀態(tài)機(jī),視圖與狀態(tài)是一一對(duì)應(yīng)的。 React實(shí)戰(zhàn)與原理筆記 概念與工具集 jsx語(yǔ)法糖;cli;state管理;jest單元測(cè)試; webpack-bundle-analyzer Sto...
摘要:因?yàn)榘姹緦⒄嬲龔U棄這三生命周期到目前為止,的渲染機(jī)制遵循同步渲染首次渲染,更新時(shí)更新時(shí)卸載時(shí)期間每個(gè)周期函數(shù)各司其職,輸入輸出都是可預(yù)測(cè),一路下來(lái)很順暢。通過(guò)進(jìn)一步觀察可以發(fā)現(xiàn),預(yù)廢棄的三個(gè)生命周期函數(shù)都發(fā)生在虛擬的構(gòu)建期間,也就是之前。 showImg(https://segmentfault.com/img/bVbweoj?w=559&h=300); 背景 前段時(shí)間準(zhǔn)備前端招聘事項(xiàng)...
摘要:一般情況下,都是作為等其他子路由的上層路由,使用了,接收一個(gè)屬性,傳遞給消費(fèi)子組件。創(chuàng)建對(duì)象,兼容老瀏覽器,其他和沒(méi)有大區(qū)別總結(jié)分為四個(gè)包,分別為,其中是瀏覽器相關(guān),是相關(guān),是核心也是共同部分,是一些配置相關(guān)。 這篇文章主要講的是分析 react-router 源碼,版本是 v5.x,以及 SPA 路由實(shí)現(xiàn)的原理。 文章首發(fā)地址 單頁(yè)面應(yīng)用都用到了路由 router,目前來(lái)看實(shí)現(xiàn)路由有...
摘要:所謂知其然還要知其所以然本文將分析的部分源碼包括組件初始渲染的過(guò)程和組件更新的過(guò)程在這之前假設(shè)讀者已經(jīng)對(duì)有一定了解知道區(qū)別了解生命周期事務(wù)批量更新大致概念等如何分析源碼代碼架構(gòu)預(yù)覽首先我們找到在上的地址把版本的源碼下來(lái)觀察它的整體架構(gòu)這 所謂知其然還要知其所以然. 本文將分析 React 15-stable的部分源碼, 包括組件初始渲染的過(guò)程和組件更新的過(guò)程.在這之前, 假設(shè)讀者已經(jīng):...
摘要:通過(guò)前端路由可以實(shí)現(xiàn)單頁(yè)應(yīng)用本文首先從前端路由的原理出發(fā),詳細(xì)介紹了前端路由原理的變遷。接著從的源碼出發(fā),深入理解是如何實(shí)現(xiàn)前端路由的。執(zhí)行上述的賦值后,頁(yè)面的發(fā)生改變。 ??react-router等前端路由的原理大致相同,可以實(shí)現(xiàn)無(wú)刷新的條件下切換顯示不同的頁(yè)面。路由的本質(zhì)就是頁(yè)面的URL發(fā)生改變時(shí),頁(yè)面的顯示結(jié)果可以根據(jù)URL的變化而變化,但是頁(yè)面不會(huì)刷新。通過(guò)前端路由可以實(shí)現(xiàn)...
閱讀 1710·2021-11-12 10:36
閱讀 1628·2021-11-12 10:36
閱讀 3452·2021-11-02 14:46
閱讀 3822·2019-08-30 15:56
閱讀 3576·2019-08-30 15:55
閱讀 1469·2019-08-30 15:44
閱讀 1056·2019-08-30 14:00
閱讀 2744·2019-08-29 18:41