欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

React?Fiber構(gòu)建completeWork源碼解析

 更新時間:2023年02月06日 09:59:38   作者:GW_劉振  
這篇文章主要為大家介紹了React?Fiber構(gòu)建completeWork源碼解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪

引言

之前我們介紹了beginWork,react使用的是深度優(yōu)先遍歷算法,整個fiber的構(gòu)建都遵循此算法。

這也意味著,并不是所有節(jié)點beginWork完成后,才去進(jìn)行completeWork。

當(dāng)beginWork的next為null時,將進(jìn)去completeWork。

一. completeUnitOfWork

function completeUnitOfWork(unitOfWork) {
  var completedWork = unitOfWork;
  do {
     // ...
     next = completeWork(current, completedWork, subtreeRenderLanes);
     // ...
     if (returnFiber !== null && (returnFiber.flags & Incomplete) === NoFlags) {
       if (returnFiber.firstEffect === null) {
          returnFiber.firstEffect = completedWork.firstEffect;
        }
        if (completedWork.lastEffect !== null) {
          if (returnFiber.lastEffect !== null) {
              returnFiber.lastEffect.nextEffect = completedWork.firstEffect;
            }
            returnFiber.lastEffect = completedWork.lastEffect;
        }
        var flags = completedWork.flags;
        if (flags > PerformedWork) {
          if (returnFiber.lastEffect !== null) {
            returnFiber.lastEffect.nextEffect = completedWork;
          }else {
            returnFiber.firstEffect = completedWork;
          }
          returnFiber.lastEffect = completedWork;
        }
     }
     var siblingFiber = completedWork.sibling;
     if (siblingFiber !== null) {
        workInProgress = siblingFiber;
        return;
     }
     completedWork = returnFiber;
     workInProgress = completedWork;
  } while (completedWork !== null);
  // ...
}

根據(jù)深度優(yōu)先算法,當(dāng)beginWork完成其中某個子樹干的最后一個節(jié)點時,進(jìn)入completeUnitOfWork。根據(jù)最后的這個節(jié)點先完成completeWork,依次往上,直到找到相鄰節(jié)點。

核心方法分為2部分,其一:completeWork,其二:完成effect掛載,并串聯(lián)所有節(jié)點的effect(包括節(jié)點內(nèi)部的effect)組裝成環(huán)狀鏈表。

二. completeWork

function completeWork(current, workInProgress, renderLanes) {
    var newProps = workInProgress.pendingProps;
    switch (workInProgress.tag) {
      case IndeterminateComponent:
      case LazyComponent:
      case SimpleMemoComponent:
      case FunctionComponent:
      case ForwardRef:
      case Fragment:
      case Mode:
      case Profiler:
      case ContextConsumer:
      case MemoComponent:
        return null;
      case ClassComponent:
        // ...
      case HostRoot:
       // ...
        updateHostContainer(workInProgress);
        return null;
      case HostComponent:
        // ...
         var instance = createInstance(type, newProps, rootContainerInstance, currentHostContext, workInProgress);
         appendAllChildren(instance, workInProgress, false, false);
         workInProgress.stateNode = instance;
        // ...
     // ...
}

這里很神奇的是,在react內(nèi)部updateHostContainer居然是個空函數(shù),也許后續(xù)版本做do something吧。 普通節(jié)點將進(jìn)入HostComponent。

createInstance

function createInstance(type, props, rootContainerInstance, hostContext, internalInstanceHandle) {
    var parentNamespace;
    {
      // TODO: take namespace into account when validating.
      var hostContextDev = hostContext;
      validateDOMNesting(type, null, hostContextDev.ancestorInfo);
      if (typeof props.children === 'string' || typeof props.children === 'number') {
        var string = '' + props.children;
        var ownAncestorInfo = updatedAncestorInfo(hostContextDev.ancestorInfo, type);
        validateDOMNesting(null, string, ownAncestorInfo);
      }
      parentNamespace = hostContextDev.namespace;
    }
    var domElement = createElement(type, props, rootContainerInstance, parentNamespace);
    precacheFiberNode(internalInstanceHandle, domElement);
    updateFiberProps(domElement, props);
    return domElement;
  }

createElement

function createElement(type, props, rootContainerElement, parentNamespace) {
  // script標(biāo)簽處理...
  // ...
  // flow相關(guān)webComponents處理
  // ...
  domElement = ownerDocument.createElement(type);
  // select標(biāo)簽特殊處理,單獨設(shè)置multiple以及size
  // 非法標(biāo)簽告警處理...
  return domElement;
}

到這里,我們可以看到開始創(chuàng)建每個fiber節(jié)點對應(yīng)的dom對象了。但是并沒有插入到文檔流中。 那么真實的dom是如何連接到fiber對象呢?

precacheFiberNode,會在每個真實dom對象下,掛載對應(yīng)的節(jié)點的fiber,precache是以_reactFiber$+隨機(jī)數(shù)的屬性。

updateFiberProps,會在每個真實dom對象下,掛載對應(yīng)的props children即element對象,以_reactProps$+隨機(jī)數(shù)。

問題來了:每個fiber對應(yīng)的真實dom對象,是單個構(gòu)建單個存儲?還是構(gòu)建一個總的dom樹?這之間是如何關(guān)聯(lián)起來的?

appendAllChildren

appendAllChildren = function (parent, workInProgress, needsVisibilityToggle, isHidden) {
    var node = workInProgress.child;
      while (node !== null) {
        if (node.tag === HostComponent || node.tag === HostText) {
          appendInitialChild(parent, node.stateNode);
        } else if (node.tag === HostPortal) ; else if (node.child !== null) {
          node.child.return = node;
          node = node.child;
          continue;
        }
        if (node === workInProgress) {
          return;
        }
        while (node.sibling === null) {
          if (node.return === null || node.return === workInProgress) {
            return;
          }
          node = node.return;
        }
        node.sibling.return = node.return;
        node = node.sibling;
      }
}

這里已經(jīng)很明顯了,根據(jù)之前創(chuàng)建的fiber鏈表,循環(huán)node節(jié)點,普通節(jié)點將調(diào)用appendInitialChild。即使用:parentInstance.appendChild(child); 至此可以看到循環(huán)結(jié)束后,生成了整個待插入的DOM節(jié)點(頁面首次渲染時)。

另外需要注意的是,根據(jù)fiber關(guān)聯(lián)dom也是在這個階段進(jìn)行的(不是dom關(guān)聯(lián)fiber)

最后如何存在ref,當(dāng)前的fiber樹對應(yīng)的flags將和Ref的二進(jìn)制數(shù)據(jù)取位運算或。(這很重要)

三. Effect

react推崇的是函數(shù)式編程,在一個函數(shù)組件里,如果存在useEffect等方法,那么react認(rèn)為這是一個副作用函數(shù)組件。那么這些副作用是如何組織起來,又是在什么階段運行的呢?

早在beginWork,fiber的flags默認(rèn)都是二進(jìn)制0,如果存在副作用,如:useEffect,ref,useLayoutEffect等等,首次將被設(shè)置為Placement。但為什么存在useEffect函數(shù)組件的fiber對象,flags都是256以上的數(shù)值?

我們以useEffect為例,一探究竟。

useEffect

function useEffect(create, deps) {
    var dispatcher = resolveDispatcher();
    return dispatcher.useEffect(create, deps);
}

沒錯,useEffect方法定義就兩行代碼。

resolveDispatcher

function resolveDispatcher() {
    var dispatcher = ReactCurrentDispatcher.current;
    if (!(dispatcher !== null)) {
      {
        throw Error( "Invalid hook call. Hooks can only be called inside of the body of a function component. This could happen for one of the following reasons:\n1. You might have mismatching versions of React and the renderer (such as React DOM)\n2. You might be breaking the Rules of Hooks\n3. You might have more than one copy of React in the same app\nSee https://reactjs.org/link/invalid-hook-call for tips about how to debug and fix this problem." );
      }
    }
    return dispatcher;
  }

我們繼續(xù)看下ReactCurrentDispatcher的定義:

const ReactCurrentDispatcher = {
  /**
   * @internal
   * @type {ReactComponent}
   */
  current: (null: null | Dispatcher),
};

最早的ReactCurrentDispatcher,是在renderRoot階段。如果root或Lane被改變了,原來的dispatch可能被置空了或首次不存在,使用當(dāng)前的ContextOnlyDispatcher替代。

在函數(shù)組件beginWork階段,在執(zhí)行函數(shù)組件生成element對象之前,會賦值HooksDispatcherOnMount,這就是dispatch。

我們來看看HooksDispatcher:

{
      readContext: function (context, observedBits) {
        return readContext(context, observedBits);
      },
      useCallback: function (callback, deps) {
        currentHookNameInDev = 'useCallback';
        mountHookTypesDev();
        checkDepsAreArrayDev(deps);
        return mountCallback(callback, deps);
      },
      useContext: function (context, observedBits) {
        currentHookNameInDev = 'useContext';
        mountHookTypesDev();
        return readContext(context, observedBits);
      },
      useEffect: function (create, deps) {
        currentHookNameInDev = 'useEffect';
        mountHookTypesDev();
        checkDepsAreArrayDev(deps);
        return mountEffect(create, deps);
      },
      useImperativeHandle: function (ref, create, deps) {
        currentHookNameInDev = 'useImperativeHandle';
        mountHookTypesDev();
        checkDepsAreArrayDev(deps);
        return mountImperativeHandle(ref, create, deps);
      },
      useLayoutEffect: function (create, deps) {
        currentHookNameInDev = 'useLayoutEffect';
        mountHookTypesDev();
        checkDepsAreArrayDev(deps);
        return mountLayoutEffect(create, deps);
      },
      useMemo: function (create, deps) {
        currentHookNameInDev = 'useMemo';
        mountHookTypesDev();
        checkDepsAreArrayDev(deps);
        var prevDispatcher = ReactCurrentDispatcher$1.current;
        ReactCurrentDispatcher$1.current = InvalidNestedHooksDispatcherOnMountInDEV;
        try {
          return mountMemo(create, deps);
        } finally {
          ReactCurrentDispatcher$1.current = prevDispatcher;
        }
      },
      useReducer: function (reducer, initialArg, init) {
        currentHookNameInDev = 'useReducer';
        mountHookTypesDev();
        var prevDispatcher = ReactCurrentDispatcher$1.current;
        ReactCurrentDispatcher$1.current = InvalidNestedHooksDispatcherOnMountInDEV;
        try {
          return mountReducer(reducer, initialArg, init);
        } finally {
          ReactCurrentDispatcher$1.current = prevDispatcher;
        }
      },
      useRef: function (initialValue) {
        currentHookNameInDev = 'useRef';
        mountHookTypesDev();
        return mountRef(initialValue);
      },
      useState: function (initialState) {
        currentHookNameInDev = 'useState';
        mountHookTypesDev();
        var prevDispatcher = ReactCurrentDispatcher$1.current;
        ReactCurrentDispatcher$1.current = InvalidNestedHooksDispatcherOnMountInDEV;
        try {
          return mountState(initialState);
        } finally {
          ReactCurrentDispatcher$1.current = prevDispatcher;
        }
      },
      useDebugValue: function (value, formatterFn) {
        currentHookNameInDev = 'useDebugValue';
        mountHookTypesDev();
        return mountDebugValue();
      },
      useDeferredValue: function (value) {
        currentHookNameInDev = 'useDeferredValue';
        mountHookTypesDev();
        return mountDeferredValue(value);
      },
      useTransition: function () {
        currentHookNameInDev = 'useTransition';
        mountHookTypesDev();
        return mountTransition();
      },
      useMutableSource: function (source, getSnapshot, subscribe) {
        currentHookNameInDev = 'useMutableSource';
        mountHookTypesDev();
        return mountMutableSource(source, getSnapshot, subscribe);
      },
      useOpaqueIdentifier: function () {
        currentHookNameInDev = 'useOpaqueIdentifier';
        mountHookTypesDev();
        return mountOpaqueIdentifier();
      },
      unstable_isNewReconciler: enableNewReconciler
    };

其中useEffect重點執(zhí)行mountEffect(create, deps)

function mountEffect(create, deps) {
    {
      // $FlowExpectedError - jest isn't a global, and isn't recognized outside of tests
      if ('undefined' !== typeof jest) {
        warnIfNotCurrentlyActingEffectsInDEV(currentlyRenderingFiber$1);
      }
    }
    return mountEffectImpl(Update | Passive, Passive$1, create, deps);
  }

mountEffectImpl

function mountEffectImpl(fiberFlags, hookFlags, create, deps) {
  var hook = mountWorkInProgressHook();
    var nextDeps = deps === undefined ? null : deps;
    currentlyRenderingFiber$1.flags |= fiberFlags;
    hook.memoizedState = pushEffect(HasEffect | hookFlags, create, undefined, nextDeps);
}

hook對象數(shù)據(jù)結(jié)構(gòu)如下:

{
    memoizedState: null,
    baseState: null,
    baseQueue: null,
    queue: null,
    next: null
};

問題來了:

  • currentlyRenderingFiber是什么?和workInProgressFiber有什么關(guān)系?
  • 多個函數(shù)組件和單個函數(shù)組件中多個Hook是如何關(guān)聯(lián)起來的?
  • 整個hook和fiber怎么關(guān)聯(lián)起來的?
  • 完整的hooks數(shù)據(jù)結(jié)構(gòu)又是什么?

currentlyRenderingFiber是當(dāng)前正在rendering階段的fiber對象,早在renderHook初始化階段賦值了workInProgressFiber。所以當(dāng)前函數(shù)組件的flags在這里被改變了,即有副作用的函數(shù)flags = flags | Update | Passive。

根據(jù)二進(jìn)制位運算,根函數(shù)組件庫flags = 518,當(dāng)然這個數(shù)值也不是固定不變的,因為變化的beginWork階段初始flags值。是根據(jù)不同的effects會有不同的初始值。

pushEffect

function pushEffect(tag, create, destroy, deps) {
    var effect = {
      tag: tag,
      create: create,
      destroy: destroy,
      deps: deps,
      // Circular
      next: null
    };
    var componentUpdateQueue = currentlyRenderingFiber$1.updateQueue;
    if (componentUpdateQueue === null) {
      componentUpdateQueue = createFunctionComponentUpdateQueue();
      currentlyRenderingFiber$1.updateQueue = componentUpdateQueue;
      componentUpdateQueue.lastEffect = effect.next = effect;
    } else {
      var lastEffect = componentUpdateQueue.lastEffect;
      if (lastEffect === null) {
        componentUpdateQueue.lastEffect = effect.next = effect;
      } else {
        var firstEffect = lastEffect.next;
        lastEffect.next = effect;
        effect.next = firstEffect;
        componentUpdateQueue.lastEffect = effect;
      }
    }
    return effect;
  }

對于tag的值,是HasEffect和Passive按位運算或的結(jié)果,實際上固定是5。

需要注意的是,函數(shù)組件的updateQueue和rootFiber的不一樣,以及普通節(jié)點的數(shù)據(jù)結(jié)構(gòu)和作用也都不一樣。 函數(shù)組件的updateQueue關(guān)聯(lián)的是effect。這和render初始化階段rootFiber有巨大的差異。

上面的代碼很簡單,每個函數(shù)組件如果存在多個effect,那么會將這些effect順序關(guān)聯(lián)起來,這個函數(shù)組件的fiberr.updateQueue對應(yīng)lastEffect,next即下一個effect,直到最后形成一個首尾相連的環(huán)狀鏈表結(jié)構(gòu)。

為什么是環(huán)狀?這個待到后續(xù)調(diào)度階段再解釋。

再思考一個問題:這里只是解決了單個組件內(nèi)的effect構(gòu)建,那么整個fiber鏈表里effect構(gòu)建是怎么樣的?執(zhí)行的順序又是什么?

四. rootFiber-Effect

在completedWork的最后,根據(jù)深度優(yōu)先遍歷算法,將每個節(jié)點的firstEffect層層往上傳遞,一直到rootFiber。而lastEffect也是層層往上判斷,直到上層最后一個effect,做為rootFiber的lastEffect。

每個fiber effect通過nextEffect鏈接起來,而fiber內(nèi)部通過updateQueue鏈接自身的effect環(huán)狀鏈表。

至此,completeWork階段就完成了,rootFiber以及各fiber節(jié)點大部分屬性都構(gòu)建完成了。

下一章,將進(jìn)入commit階段,更多關(guān)于React Fiber構(gòu)建completeWork的資料請關(guān)注腳本之家其它相關(guān)文章!

以上就是React Fiber構(gòu)建completeWork源碼解析的詳細(xì)內(nèi)容,更多關(guān)于React Fiber構(gòu)建completeWork的資料請關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • React中如何使用scss

    React中如何使用scss

    這篇文章主要介紹了React中如何使用scss問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2023-03-03
  • React Hooks使用常見的坑

    React Hooks使用常見的坑

    React Hooks 是 React 16.8 引入的新特性,允許我們在不使用 Class 的前提下使用 state 和其他特性。接下來通過本文給大家分享React Hooks使用避坑指南,一起學(xué)習(xí)下吧
    2021-06-06
  • 一文詳解React中Hook的作用和用處

    一文詳解React中Hook的作用和用處

    Hook是一種函數(shù),它可以讓你在函數(shù)組件中“掛鉤”到React狀態(tài)和生命周期等特性,這篇文章主要和大家介紹了React中Hook的作用和用處,希望對大家有所幫助
    2023-05-05
  • 每天學(xué)習(xí)一個hooks?useMount

    每天學(xué)習(xí)一個hooks?useMount

    這篇文章主要為大家介紹了每天學(xué)習(xí)一個hooks?useMount,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-05-05
  • 如何深入理解React的ref 屬性

    如何深入理解React的ref 屬性

    關(guān)于 Refs ,React 官網(wǎng)講解的對于新手來說不太友好,還是自己一字一句解讀后并以代碼驗證的方式后真正理解的.
    2021-05-05
  • React中的Hooks路由跳轉(zhuǎn)問題

    React中的Hooks路由跳轉(zhuǎn)問題

    這篇文章主要介紹了React中的Hooks路由跳轉(zhuǎn)問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-12-12
  • 深入理解React高階組件

    深入理解React高階組件

    本篇文章主要介紹了深入理解React高階組件,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-09-09
  • react實現(xiàn)自定義拖拽hook

    react實現(xiàn)自定義拖拽hook

    這篇文章主要為大家詳細(xì)介紹了react實現(xiàn)自定義拖拽hook,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2022-08-08
  • 一文詳解如何React中實現(xiàn)插槽

    一文詳解如何React中實現(xiàn)插槽

    這篇文章主要為大家詳細(xì)介紹了如何在React中實現(xiàn)插槽,文中的示例代碼講解詳細(xì),對我們的學(xué)習(xí)或工作具有一定的借鑒價值,需要的可以了解一下
    2023-03-03
  • React各種狀態(tài)管理器的解讀及使用方法

    React各種狀態(tài)管理器的解讀及使用方法

    這篇文章主要介紹了對于React各種狀態(tài)管理器的解讀,文中給大家提到了狀態(tài)管理器是如何使用的,本文通過實例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2021-12-12

最新評論