摘要:為了幫助理解,我們繼續(xù)加日志司徒正美,加群一起研究與只要收到更新對(duì)象,就會(huì)被調(diào)度程序調(diào)用。渲染器在將來(lái)的某個(gè)時(shí)刻調(diào)用。導(dǎo)步肯定為歡迎加繼續(xù)略也是怒長(zhǎng),代碼的特點(diǎn)是許多巨型類,巨型方法,有之遺風(fēng)。
insertUpdateIntoFiber 會(huì)根據(jù)fiber的狀態(tài)創(chuàng)建一個(gè)或兩個(gè)列隊(duì)對(duì)象,對(duì)象是長(zhǎng)成這樣的
//by 司徒正美, 加群:370262116 一起研究React與anujs // https://github.com/RubyLouvre/anu 歡迎加star function createUpdateQueue(baseState) {//我們現(xiàn)在是丟了一個(gè)null做傳參 var queue = { baseState: baseState, expirationTime: NoWork,//NoWork會(huì)被立即執(zhí)行 first: null, last: null, callbackList: null, hasForceUpdate: false, isInitialized: false }; return queue; }
scheduleWork是一個(gè)奇怪的方法,只是添加一下參數(shù)
function scheduleWork(fiber, expirationTime) { return scheduleWorkImpl(fiber, expirationTime, false); }
scheduleWorkImpl的最開(kāi)頭有一個(gè)recordScheduleUpdate方法,用來(lái)記錄調(diào)度器的執(zhí)行狀態(tài),如注釋所示,它現(xiàn)在相當(dāng)于什么都沒(méi)有做
function recordScheduleUpdate() { if (enableUserTimingAPI) {//全局變量,默認(rèn)為true if (isCommitting) {//全局變量,默認(rèn)為false, 沒(méi)有進(jìn)入分支 hasScheduledUpdateInCurrentCommit = true; } //全局變量,默認(rèn)為null,沒(méi)有沒(méi)有進(jìn)入分支 if (currentPhase !== null && currentPhase !== "componentWillMount" && currentPhase !== "componentWillReceiveProps") { hasScheduledUpdateInCurrentPhase = true; } } }
scheduleWorkImpl的一些分支非常復(fù)雜,我們打一些斷點(diǎn)
function computeExpirationForFiber(fiber) { var expirationTime = void 0; if (expirationContext !== NoWork) { // An explicit expiration context was set; expirationTime = expirationContext; } else if (isWorking) { if (isCommitting) { // Updates that occur during the commit phase should have sync priority // by default. expirationTime = Sync; } else { // Updates during the render phase should expire at the same time as // the work that is being rendered. expirationTime = nextRenderExpirationTime; } } else { // No explicit expiration context was set, and we"re not currently // performing work. Calculate a new expiration time. if (useSyncScheduling && !(fiber.internalContextTag & AsyncUpdates)) { // This is a sync update console.log("expirationTime", Sync) expirationTime = Sync;//命中這里 } else { // This is an async update expirationTime = computeAsyncExpiration(); } } return expirationTime; } function checkRootNeedsClearing(root, fiber, expirationTime) { if (!isWorking && root === nextRoot && expirationTime < nextRenderExpirationTime) { console.log("checkRootNeedsClearing對(duì)nextRoot,nextUnitOfWork,nextRenderExpirationTime進(jìn)行置空") // Restart the root from the top. if (nextUnitOfWork !== null) { // This is an interruption. (Used for performance tracking.) interruptedBy = fiber; } nextRoot = null; nextUnitOfWork = null; nextRenderExpirationTime = NoWork; }else{ console.log("checkRootNeedsClearing就是想醬油") } } function scheduleWorkImpl(fiber, expirationTime, isErrorRecovery) { recordScheduleUpdate();//現(xiàn)在什么也沒(méi)做 var node = fiber; while (node !== null) { // Walk the parent path to the root and update each node"s // expiration time. if (node.expirationTime === NoWork || node.expirationTime > expirationTime) { node.expirationTime = expirationTime;//由于默認(rèn)就是NoWork,因此會(huì)被重寫(xiě) Sync } if (node.alternate !== null) {//這里進(jìn)不去 if (node.alternate.expirationTime === NoWork || node.alternate.expirationTime > expirationTime) { node.alternate.expirationTime = expirationTime; } } if (node["return"] === null) { if (node.tag === HostRoot) {//進(jìn)入這里 var root = node.stateNode; checkRootNeedsClearing(root, fiber, expirationTime); console.log("requestWork",root, expirationTime) requestWork(root, expirationTime); checkRootNeedsClearing(root, fiber, expirationTime); } else { return; } } node = node["return"]; } }
輸出如下
requestWork也很難理解,里面太多全局變量,覺(jué)得不是前端的人搞的。為了幫助理解,我們繼續(xù)加日志
//by 司徒正美, 加群:370262116 一起研究React與anujs // requestWork is called by the scheduler whenever a root receives an update. // It"s up to the renderer to call renderRoot at some point in the future. /* 只要root收到更新(update對(duì)象),requestWork就會(huì)被調(diào)度程序調(diào)用。 渲染器在將來(lái)的某個(gè)時(shí)刻調(diào)用renderRoot。 */ function requestWork(root, expirationTime) { if (nestedUpdateCount > NESTED_UPDATE_LIMIT) { invariant_1(false, "Maximum update depth exceeded. This can happen when a component repeatedly calls setState inside componentWillUpdate or componentDidUpdate. React limits the number of nested updates to prevent infinite loops."); } // Add the root to the schedule. // Check if this root is already part of the schedule. if (root.nextScheduledRoot === null) { // This root is not already scheduled. Add it. console.log("設(shè)置remainingExpirationTime",expirationTime) root.remainingExpirationTime = expirationTime; if (lastScheduledRoot === null) { console.log("設(shè)置firstScheduledRoot, lastScheduledRoot") firstScheduledRoot = lastScheduledRoot = root; root.nextScheduledRoot = root; } else { lastScheduledRoot.nextScheduledRoot = root; lastScheduledRoot = root; lastScheduledRoot.nextScheduledRoot = firstScheduledRoot; } } else { // This root is already scheduled, but its priority may have increased. var remainingExpirationTime = root.remainingExpirationTime; if (remainingExpirationTime === NoWork || expirationTime < remainingExpirationTime) { // Update the priority. root.remainingExpirationTime = expirationTime; } } if (isRendering) { // Prevent reentrancy. Remaining work will be scheduled at the end of // the currently rendering batch. return; } if (isBatchingUpdates) { // Flush work at the end of the batch. if (isUnbatchingUpdates) { // ...unless we"re inside unbatchedUpdates, in which case we should // flush it now. nextFlushedRoot = root; nextFlushedExpirationTime = Sync; console.log("performWorkOnRoot") performWorkOnRoot(nextFlushedRoot, nextFlushedExpirationTime); } return; } // TODO: Get rid of Sync and use current time? if (expirationTime === Sync) { console.log("進(jìn)入performWork") performWork(Sync, null); } else { scheduleCallbackWithExpiration(expirationTime); } }
從日志輸出來(lái)看,requestWork只是修改了兩個(gè)全局變量,然后進(jìn)入performWork。這三個(gè)內(nèi)部方法起名很有意思。scheduleWork意為打算工作,requestWork意為申請(qǐng)工作,performWork意為努力工作(正式上班)
function performWork(minExpirationTime, dl) { deadline = dl; // Keep working on roots until there"s no more work, or until the we reach // the deadline. //這里會(huì)將root設(shè)置為highestPriorityRoot findHighestPriorityRoot(); if (enableUserTimingAPI && deadline !== null) { var didExpire = nextFlushedExpirationTime < recalculateCurrentTime(); console.log(didExpire) stopRequestCallbackTimer(didExpire); } while (nextFlushedRoot !== null && nextFlushedExpirationTime !== NoWork && (minExpirationTime === NoWork || nextFlushedExpirationTime <= minExpirationTime) && !deadlineDidExpire) { console.log("performWorkOnRoot") performWorkOnRoot(highestPriorityRoot, nextFlushedExpirationTime); // Find the next highest priority work. findHighestPriorityRoot(); } // We"re done flushing work. Either we ran out of time in this callback, // or there"s no more work left with sufficient priority. // If we"re inside a callback, set this to false since we just completed it. if (deadline !== null) { callbackExpirationTime = NoWork; callbackID = -1; } // If there"s work left over, schedule a new callback. if (nextFlushedExpirationTime !== NoWork) { console.log("scheduleCallbackWithExpiration") scheduleCallbackWithExpiration(nextFlushedExpirationTime); } // Clean-up. deadline = null; deadlineDidExpire = false; nestedUpdateCount = 0; if (hasUnhandledError) { //如果有沒(méi)處理的錯(cuò)誤則throw var _error4 = unhandledError; unhandledError = null; hasUnhandledError = false; throw _error4; } }
我們終于進(jìn)入performWorkOnRoot,performWorkOnRoot的作用是區(qū)分同步渲染還是異步渲染,expirationTime等于1,因此進(jìn)入同步。導(dǎo)步肯定為false
// https://github.com/RubyLouvre/anu 歡迎加star function performWorkOnRoot(root, expirationTime) { isRendering = true; // Check if this is async work or sync/expired work. // TODO: Pass current time as argument to renderRoot, commitRoot if (expirationTime <= recalculateCurrentTime()) { // Flush sync work. var finishedWork = root.finishedWork; console.log("Flush sync work.", finishedWork) if (finishedWork !== null) { // This root is already complete. We can commit it. root.finishedWork = null; console.log("commitRoot") root.remainingExpirationTime = commitRoot(finishedWork); } else { root.finishedWork = null; console.log("renderRoot") finishedWork = renderRoot(root, expirationTime); if (finishedWork !== null) { console.log("繼續(xù)commitRoot") // We"ve completed the root. Commit it. root.remainingExpirationTime = commitRoot(finishedWork); } } } else { console.log("Flush async work.") // Flush async work. // ...略 } isRendering = false; }
renderRoot也是怒長(zhǎng),React16代碼的特點(diǎn)是許多巨型類,巨型方法,有JAVA之遺風(fēng)。renderRoot只有前面幾行是可能處理虛擬DOM(或叫fiber),后面都是錯(cuò)誤邊界的
function renderRoot(root, expirationTime) { isWorking = true; // We"re about to mutate the work-in-progress tree. If the root was pending // commit, it no longer is: we"ll need to complete it again. root.isReadyForCommit = false; // Check if we"re starting from a fresh stack, or if we"re resuming from // previously yielded work. if (root !== nextRoot || expirationTime !== nextRenderExpirationTime || nextUnitOfWork === null) { // Reset the stack and start working from the root. resetContextStack(); nextRoot = root; nextRenderExpirationTime = expirationTime; //可能是用來(lái)工作的代碼 console.log("createWorkInProgress") nextUnitOfWork = createWorkInProgress(nextRoot.current, null, expirationTime); } //可能是用來(lái)工作的代碼 console.log("startWorkLoopTimer") startWorkLoopTimer(nextUnitOfWork); // 處理錯(cuò)誤邊界 var didError = false; var error = null; invokeGuardedCallback$1(null, workLoop, null, expirationTime); // An error was thrown during the render phase. while (didError) { console.log("componentDidCatch的相關(guān)實(shí)現(xiàn)") if (didFatal) { // This was a fatal error. Don"t attempt to recover from it. firstUncaughtError = error; break; } var failedWork = nextUnitOfWork; if (failedWork === null) { // An error was thrown but there"s no current unit of work. This can // happen during the commit phase if there"s a bug in the renderer. didFatal = true; continue; } // 處理錯(cuò)誤邊界 var boundary = captureError(failedWork, error); !(boundary !== null) ? invariant_1(false, "Should have found an error boundary. This error is likely caused by a bug in React. Please file an issue.") : void 0; if (didFatal) { // The error we just captured was a fatal error. This happens // when the error propagates to the root more than once. continue; } // 處理錯(cuò)誤邊界 didError = false; error = null; // We"re finished working. Exit the error loop. break; } // 處理錯(cuò)誤邊界 var uncaughtError = firstUncaughtError; // We"re done performing work. Time to clean up. stopWorkLoopTimer(interruptedBy); interruptedBy = null; isWorking = false; didFatal = false; firstUncaughtError = null; // 處理錯(cuò)誤邊界 if (uncaughtError !== null) { onUncaughtError(uncaughtError); } return root.isReadyForCommit ? root.current.alternate : null; } function resetContextStack() { // Reset the stack reset$1(); // Reset the cursors resetContext(); resetHostContainer(); } function reset$1() { console.log("reset",index) while (index > -1) { valueStack[index] = null; { fiberStack[index] = null; } index--; } } function resetContext() { consoel.log("resetContext") previousContext = emptyObject_1; contextStackCursor.current = emptyObject_1; didPerformWorkStackCursor.current = false; } function resetHostContainer() { console.log("resetHostContainer",contextStackCursor, rootInstanceStackCursor, NO_CONTEXT ) contextStackCursor.current = NO_CONTEXT; rootInstanceStackCursor.current = NO_CONTEXT; }
createWorkInProgress就是將根組件的fiber對(duì)象再?gòu)?fù)制一份,變成其alternate屬性。因此 將虛擬DOM轉(zhuǎn)換為真實(shí)DOM的重任就交給invokeGuardedCallback
var invokeGuardedCallback = function (name, func, context, a, b, c, d, e, f) { ReactErrorUtils._hasCaughtError = false; ReactErrorUtils._caughtError = null; var funcArgs = Array.prototype.slice.call(arguments, 3); try { func.apply(context, funcArgs); } catch (error) { ReactErrorUtils._caughtError = error; ReactErrorUtils._hasCaughtError = true; } //這下面還有怒長(zhǎng)(100-150L )的關(guān)于錯(cuò)誤邊界的處理,略過(guò) };
func為workLoop
//by 司徒正美, 加群:370262116 一起研究React與anujs function workLoop(expirationTime) { if (capturedErrors !== null) { // If there are unhandled errors, switch to the slow work loop. // TODO: How to avoid this check in the fast path? Maybe the renderer // could keep track of which roots have unhandled errors and call a // forked version of renderRoot. slowWorkLoopThatChecksForFailedWork(expirationTime); return; } if (nextRenderExpirationTime === NoWork || nextRenderExpirationTime > expirationTime) { return; } if (nextRenderExpirationTime <= mostRecentCurrentTime) { // Flush all expired work. while (nextUnitOfWork !== null) { console.log("performUnitOfWork",nextUnitOfWork) nextUnitOfWork = performUnitOfWork(nextUnitOfWork); } } else { // Flush asynchronous work until the deadline runs out of time. while (nextUnitOfWork !== null && !shouldYield()) { nextUnitOfWork = performUnitOfWork(nextUnitOfWork); } } }
我們終于看到工作的代碼了。 這個(gè)nextUnitOfWork 是renderRoot生成的
performUnitOfWork與beginWork的代碼,里面會(huì)根據(jù)fiber的tag進(jìn)入各種操作
//by 司徒正美, 加群:370262116 一起研究React與anujs // https://github.com/RubyLouvre/anu 歡迎加star function performUnitOfWork(workInProgress) { // The current, flushed, state of this fiber is the alternate. // Ideally nothing should rely on this, but relying on it here // means that we don"t need an additional field on the work in // progress. var current = workInProgress.alternate; // See if beginning this work spawns more work. startWorkTimer(workInProgress); { ReactDebugCurrentFiber.setCurrentFiber(workInProgress); } console.log("beginWork") var next = beginWork(current, workInProgress, nextRenderExpirationTime); { ReactDebugCurrentFiber.resetCurrentFiber(); } if (true && ReactFiberInstrumentation_1.debugTool) { ReactFiberInstrumentation_1.debugTool.onBeginWork(workInProgress); } if (next === null) { console.log("next") // If this doesn"t spawn new work, complete the current work. next = completeUnitOfWork(workInProgress); } ReactCurrentOwner.current = null; return next; } function beginWork(current, workInProgress, renderExpirationTime) { if (workInProgress.expirationTime === NoWork || workInProgress.expirationTime > renderExpirationTime) { return bailoutOnLowPriority(current, workInProgress); } switch (workInProgress.tag) { case IndeterminateComponent: return mountIndeterminateComponent(current, workInProgress, renderExpirationTime); case FunctionalComponent: return updateFunctionalComponent(current, workInProgress); case ClassComponent: return updateClassComponent(current, workInProgress, renderExpirationTime); case HostRoot: return updateHostRoot(current, workInProgress, renderExpirationTime); case HostComponent: return updateHostComponent(current, workInProgress, renderExpirationTime); case HostText: return updateHostText(current, workInProgress); case CallHandlerPhase: // This is a restart. Reset the tag to the initial phase. workInProgress.tag = CallComponent; // Intentionally fall through since this is now the same. case CallComponent: return updateCallComponent(current, workInProgress, renderExpirationTime); case ReturnComponent: // A return component is just a placeholder, we can just run through the // next one immediately. return null; case HostPortal: return updatePortalComponent(current, workInProgress, renderExpirationTime); case Fragment: return updateFragment(current, workInProgress); default: invariant_1(false, "Unknown unit of work tag. This error is likely caused by a bug in React. Please file an issue."); } }
我們?cè)僬{(diào)查一下workInProgress.tag是什么
https://github.com/facebook/r...
這里有全部fiber節(jié)點(diǎn)的類型描述,我們創(chuàng)建一個(gè)對(duì)象
// https://github.com/RubyLouvre/anu 歡迎加star var mapBeginWork = { 3: "HostRoot 根組件", 0: "IndeterminateComponent 只知道type為函數(shù)", 2: "ClassComponent 普通類組件" , 5: "HostComponent 元素節(jié)點(diǎn)", 6: "HostText 文本節(jié)點(diǎn)" } function beginWork(current, workInProgress, renderExpirationTime) { if (workInProgress.expirationTime === NoWork || workInProgress.expirationTime > renderExpirationTime) { return bailoutOnLowPriority(current, workInProgress); } console.log(workInProgress.tag, mapBeginWork[workInProgress.tag]) switch (workInProgress.tag) { //略 } }
總結(jié)一下到現(xiàn)在的所有流程:
好了,今天就這么多,想必大家都累了,下一篇繼續(xù)
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/107182.html
摘要:司徒正美,加群一起研究與用于調(diào)整渲染順序,高優(yōu)先級(jí)的組件先執(zhí)行這只是一部分更新邏輯,簡(jiǎn)直沒(méi)完沒(méi)了,下次繼續(xù)添上流程圖,回憶一下本文學(xué)到的東西 React16真是一天一改,如果現(xiàn)在不看,以后也很難看懂了。 在React16中,雖然也是通過(guò)JSX編譯得到一個(gè)虛擬DOM對(duì)象,但對(duì)這些虛擬DOM對(duì)象的再加工則是經(jīng)過(guò)翻天覆地的變化。我們需要追根溯底,看它是怎么一步步轉(zhuǎn)換過(guò)來(lái)的。我們先不看什么組件...
摘要:引言于發(fā)布版本,時(shí)至今日已更新到,且引入了大量的令人振奮的新特性,本文章將帶領(lǐng)大家根據(jù)更新的時(shí)間脈絡(luò)了解的新特性。其作用是根據(jù)傳遞的來(lái)更新。新增等指針事件。 1 引言 于 2017.09.26 Facebook 發(fā)布 React v16.0 版本,時(shí)至今日已更新到 React v16.6,且引入了大量的令人振奮的新特性,本文章將帶領(lǐng)大家根據(jù) React 更新的時(shí)間脈絡(luò)了解 React1...
摘要:因?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òu)理解引用原文是核心算法正在進(jìn)行的重新實(shí)現(xiàn)。構(gòu)建的過(guò)程就是的過(guò)程,通過(guò)來(lái)調(diào)度執(zhí)行一組任務(wù),每完成一個(gè)任務(wù)后回來(lái)看看有沒(méi)有插隊(duì)的更緊急的,把時(shí)間控制權(quán)交還給主線程,直到下一次回調(diào)再繼續(xù)構(gòu)建。 React Fiber 架構(gòu)理解 引用原文:React Fiber ArchitectureReact Fiber is an ongoing reimplementation of Reacts...
閱讀 3167·2021-11-22 09:34
閱讀 2806·2021-09-22 15:28
閱讀 835·2021-09-10 10:51
閱讀 1865·2019-08-30 14:22
閱讀 2332·2019-08-30 14:17
閱讀 2746·2019-08-30 11:01
閱讀 2306·2019-08-29 17:19
閱讀 3674·2019-08-29 13:17