免费观看又色又爽又黄的小说免费_美女福利视频国产片_亚洲欧美精品_美国一级大黄大色毛片

Reactref的原理和應用

本篇內容介紹了“React ref的原理和應用”的有關知識,在實際案例的操作過程中,不少人都會遇到這樣的困境,接下來就讓小編帶領大家學習一下如何處理這些情況吧!希望大家仔細閱讀,能夠學有所成!

成都創新互聯是專業的黎川網站建設公司,黎川接單;提供成都做網站、網站制作,網頁設計,網站設計,建網站,PHP網站建設等專業做網站服務;采用PHP框架,可快速的進行黎川網站開發網頁制作和功能擴展;專業做搜索引擎喜愛的網站,專業的做網站團隊,希望更多企業前來合作!

提到 ref或者 refs 如果你用過React 16以前的版本 第一印象都是用來訪問DOM或者修改組件實例的,

正如官網所介紹的這樣:

React ref的原理和應用

然后到了React 16.3出現的 createRef 以及16.8 hooks中的  useRef出現時,發現這里的ref好像不僅僅只有之前的綁定到DOM/組件實例的 作用?本文將帶你逐一梳理這些知識點,并嘗試分析相關源碼。

前置知識

這部分知識點不是本文重點,每個點展開都非常龐大,了方便本文理解先在這里簡單提及。

Fiber架構

Fiber是React更新時的最小單元,是一種包含指針的數據結構,從數據結構上看Fiber架構 ≈ 樹 + 鏈表。

Fiber單元是從 jsx createElement之后根據ReactElement生成的,相比  ReactElement,Fiber單元具備動態工作能力。

React 的工作流程

使用chrome perfomance錄制一個react應用渲染看函數調用棧會看到下面這張圖

React ref的原理和應用

這三塊內容分別代表: 1.生成react root節點 2.reconciler 協調生成需要更新的子節點 3.將節點更新commit 到視圖

Hooks基礎知識

在函數組件中每執行一次use開頭的hook函數都會生成一個hook對象。

type Hook = {   memoizedState: any,   // 上次更新之后的最終狀態值   queue: UpdateQueue, //更新隊列   next, // 下一個 hook 對象 };

其中memoizedState會保存該hook上次更新之后的最終狀態,比如當我們使用一次useState之后就會在memoizedState中保存初始值。

React 中大部分 hook 分為兩個階段:第一次初始化時`mount`階段和更新`update`時階段

hooks函數的執行分兩個階段 mount和 update,比如 useState只會在初始化時執行一次,下文中將提到的

useImperativeHandle 和 useRef也包括在內。

調試源碼

本文已梳理摘取了源碼相關的函數,但你如果配合源碼調試一起食用效果會更加。

本文基于React v17.0.2。

拉取React代碼并安裝依賴

將react,scheduler以及react-dom打包為commonjs

yarn build react/index,react-dom/index,scheduler --type NODE

3.進入build/node_modules/react/cjs 執行yarn link 同理 react-dom

4.在  build/node_modules/react/cjs/react.development.js中加入link標記console以確保檢查link狀態

5.使用create-react-app創建一個測試應用 并link react,react-dom

ref prop

組件上的ref屬性是一個保留屬性,你不能把ref當成一個普通的prop屬性在一個組件中獲取,比如:

const Parent = () => {     return <Child ref={{test:1}}> } const Child = (props) => {   console.log(props);   // 這里獲取不到ref屬性     return <div></div> }

這個ref去哪里了呢, React本身又對它做了什么呢?

我們知道React的解析是從createElement開始的,找到了下面創建ReactElement的地方,確實有對ref保留屬性的處理。

export function createElement(type, config, children) { let propName;   // Reserved names are extracted   const props = {};   let ref = null;   if (config != null) {     if (hasValidRef(config)) {       ref = config.ref;     }     for (propName in config) {       if (         hasOwnProperty.call(config, propName) &&         !RESERVED_PROPS.hasOwnProperty(propName)       ) {         props[propName] = config[propName];       }     }   }   return ReactElement(     type,     key,     ref,     props,     ...   ); }

從createElement開始就已經創建了對ref屬性的引用。

createElement之后我們需要構建Fiber工作樹,接下來主要講對ref相關的處理。

React對于不同的組件有不通的處理

先主要關注 FunctionComponent/ClassComponent/HostComponent(原生html標簽)

FunctionComponent

function updateFunctionComponent(current, workInProgress, Component, nextProps, renderLanes) {       try {         nextChildren = renderWithHooks(current, workInProgress, Component, nextProps, context, renderLanes);       } finally {         reenableLogs();       }       reconcileChildren(current, workInProgress, nextChildren, renderLanes);       return workInProgress.child; } functin renderWithHooks(current, workInProgress, Component, props, secondArg, nextRenderLanes){             children = Component(props, secondArg); // 這里的Component就是指我們的函數組件                 return children; }

我們可以看到函數組件在渲染的時候就是直接執行。

Class組件和原生標簽的ref prop

ClassComponent

function updateClassComponent(current, workInProgress, Component, nextProps, renderLanes) {   ...   {     ...     constructClassInstance(workInProgress, Component, nextProps);         ....   }   var nextUnitOfWork = finishClassComponent(current, workInProgress, Component, shouldUpdate, hasContext, renderLanes);     ...   return nextUnitOfWork; } function constructClassInstance(workInProgress, ctor, props) {     ....   var instance = new ctor(props, context);   // 把instance實例掛載到workInProgress stateNode屬性上   adoptClassInstance(workInProgress, instance);     .....   return instance; } function finishClassComponent(current, workInProgress, Component, shouldUpdate, hasContext, renderLanes) {   // 標記是否有ref更新   markRef(current, workInProgress); } function markRef(current, workInProgress) {   var ref = workInProgress.ref;   if (current === null && ref !== null || current !== null && current.ref !== ref) {     // Schedule a Ref effect     workInProgress.flags |= Ref;   } }

ClassComponent則是通過構造函數生成實例并標記了ref屬性。

回顧一下之前提到的React工作流程,既然是要將組件實例或者真實DOM賦值給ref那肯定不能在一開始就處理這個ref,而是根據標記到commit階段再給ref賦值。

function commitLayoutEffectOnFiber(finishedRoot, current, finishedWork, committedLanes) {     ....   {     if (finishedWork.flags & Ref) {       commitAttachRef(finishedWork);     }   }   .... } function commitAttachRef(finishedWork) {   var ref = finishedWork.ref;   if (ref !== null) {     var instance = finishedWork.stateNode;     var instanceToUse;     switch (finishedWork.tag) {       case HostComponent:         // getPublicInstance 這里調用了DOM API 返回了DOM對象         instanceToUse = getPublicInstance(instance);         break;       default:         instanceToUse = instance;     }      // 對函數回調形式設置ref的處理     if (typeof ref === 'function') {       {         ref(instanceToUse);       }     } else {       ref.current = instanceToUse;     }   } }

在commit階段,如果是原生標簽則將真實DOM賦值給ref對象的current屬性, 如果是class componnet  則是組件instance。

函數組件的ref prop

如果你對function組件未做處理直接加上ref,react會直接忽略并在開發環境給出警告

React ref的原理和應用

函數組件沒有實例可以賦值給ref對象,而且組件上的ref prop會被當作保留屬性無法在組件中獲取,那該怎么辦呢?

forwardRef

React提供了一個forwardRef函數 來處理函數組件的 ref prop,用起來就像下面這個示例:

const Parent = () => {     const childRef = useRef(null)   return <Child ref={childRef}/> } const Child = forWardRef((props,ref) => {     return <div>Child</div> }}

這個方法的源碼主體也非常簡單,返回了一個新的elementType對象,這個對象的render屬性包含了原本的這個函數組件,而$$typeof則標記了這個特殊組件類型。

function forwardRef(render) {   ....   var elementType = {     $$typeof: REACT_FORWARD_REF_TYPE,     render: render   }   ....   return elementType;  }

那么React對forwardRef這個特殊的組件是怎么處理的呢

function beginWork(current, workInProgress, renderLanes) {     ...   switch (workInProgress.tag) {     case FunctionComponent:       {        ...         return updateFunctionComponent(current, workInProgress, _Component, resolvedProps, renderLanes);       }     case ClassComponent:       {                 ....         return updateClassComponent(current, workInProgress, _Component2, _resolvedProps, renderLanes);       }     case HostComponent:       return updateHostComponent(current, workInProgress, renderLanes);     case ForwardRef:       {                 ....         // 第三個參數type就是forwardRef創建的elementType         return updateForwardRef(current, workInProgress, type, _resolvedProps2, renderLanes);       } }    function updateForwardRef(current, workInProgress, Component, nextProps, renderLanes) {     ....   var render = Component.render;   var ref = workInProgress.ref; // The rest is a fork of updateFunctionComponent   var nextChildren;   {         ...     //  將ref引用傳入renderWithHooks     nextChildren = renderWithHooks(current, workInProgress, render, nextProps, ref, renderLanes);     ...   }   workInProgress.flags |= PerformedWork;   reconcileChildren(current, workInProgress, nextChildren, renderLanes);   return workInProgress.child; }

可以看到和上面 FunctionComponent的主要區別僅僅是把ref保留屬性當成普通屬性傳入 renderWithHooks方法!

那么又有一個問題出現了,如果只是傳了一個ref引用,而沒有像Class組件那樣可以attach的實例,豈不是沒有辦法操作子函數組件的行為?

用上面的例子驗證一下

const Parent = () => {     const childRef = useRef(null)   useEffect(()=>{     console.log(childref) // { current:null }   })   return <Child ref={childRef}/> } const Child = forwardRef((props,ref) => {     return <div>Child</div> }}                            const Parent = () => {    const childRef = useRef(null)   useEffect(()=>{     console.log(childref) // { current: div }   })   return <Child ref={childRef}/> } const Child = forwardRef((props,ref) => {     return <div ref={ref}>Child</div> }}

結合輸出可以看出如果單獨使用forwardRef僅僅只能轉發ref屬性。如果ref最終沒有綁定到一個ClassCompnent或者原生DOM上那么這個ref將不會改變。

假設一個業務場景,你封裝了一個表單組件,想對外暴露一些接口比如說提交的action以及校驗等操作,這樣應該如何處理呢?

useImperativeHandle

react為我們提供了這個hook來幫助函數組件向外部暴露屬性

先看下效果

const Parent = () => {     const childRef = useRef(null)   useEffect(()=>{     chilRef.current.sayName();// child   })   return <Child ref={childRef}/> } const Child = forwardRef((props,ref) => {   useImperativeHandle(ref,()=>({     sayName:()=>{         console.log('child')     }   }))     return <div>Child</div> }}

看一下該hook的源碼部分(以hook mount階段為例):

useImperativeHandle: function (ref, create, deps) {       currentHookNameInDev = 'useImperativeHandle';       mountHookTypesDev();       checkDepsAreArrayDev(deps);       return mountImperativeHandle(ref, create, deps);  } function mountImperativeHandle(ref, create, deps) {   {     if (typeof create !== 'function') {       error('Expected useImperativeHandle() second argument to be a function ' + 'that creates a handle. Instead received: %s.', create !== null ? typeof create : 'null');     }   } // TODO: If deps are provided, should we skip comparing the ref itself?   var effectDeps = deps !== null && deps !== undefined ? deps.concat([ref]) : null;   var fiberFlags = Update;   return mountEffectImpl(fiberFlags, Layout, imperativeHandleEffect.bind(null, create, ref), effectDeps); } function imperativeHandleEffect(create, ref) {   if (typeof ref === 'function') {     var refCallback = ref;     var _inst = create();     refCallback(_inst);     return function () {       refCallback(null);     };   } else if (ref !== null && ref !== undefined) {     var refObject = ref;     {       if (!refObject.hasOwnProperty('current')) {         error('Expected useImperativeHandle() first argument to either be a ' + 'ref callback or React.createRef() object. Instead received: %s.', 'an object with keys {' + Object.keys(refObject).join(', ') + '}');       }     }         // 這里執行了傳給hook的第二個參數     var _inst2 = create();     refObject.current = _inst2;     return function () {       refObject.current = null;     };   } }

其實就是將我們需要暴露的對象及傳給useImperativeHandle的第二個函數參數執行結果賦值給了ref的current對象。

同一份引用

到此為止我們大致梳理了組件上ref prop 的工作流程,以及如何在函數組件中使用ref prop,貌似比想象中簡單。

上面的過程我們注意到從createElement再到構建WorkInProgess  Fiber樹到最后commit的過程,ref似乎是一直在被傳遞。

中間過程的代碼過于龐大復雜,但是我們可以通過一個簡單的測試來驗證一下。

const isEqualRefDemo = () => {     const isEqualRef = useRef(1)   return <input key="test" ref={isEqualRef}> }

對于 class component 和 原生標簽來說 就是 createElement 到 commitAttachRef之前:

React ref的原理和應用

React ref的原理和應用

在createElement里將ref掛載給window對象,然后在commitAttachRef里判斷一下這兩次的ref是否全等。

React ref的原理和應用

對于函數組件來說就是 createElement 到 hook執行 imperativeHandleEffect 之前:

const Parent = () => {     const childRef = useRef(1)   useEffect(()=>{     chilRef.current.sayName();// child   })   return <Child ref={childRef}/> } const Child = forwardRef((props,ref) => {   useImperativeHandle(ref,()=>({     sayName:()=>{         console.log('child')     }   }))     return <div>Child</div> }}

 React ref的原理和應用

React ref的原理和應用

從createElement添加ref到React整個渲染過程的末尾(commit階段)被賦值前,這個ref都是同一份引用。

這也正如 ref單詞的本意 reference引用一樣。

小節總結

1.ref出現在組件上時是一個保留屬性

2.ref在組件存在的生命周期內維護了同一個引用(可變對象 MutableObject)

3.當ref掛載的對象是原生html標簽時會ref對象的current屬性會被賦值為真實DOM  而如果是React組件會被賦值為React"組件實例"

4.ref掛載都在commit階段處理

創建ref的方式

ref prop相當于在組件上挖了一個“坑” 來承接 ref對象,但是這樣還不夠我們還需要先創建ref對象

字符串ref & callback ref

這兩種創建ref的方式不再贅述,官網以及社區優秀文章可供參考。

https://zh-hans.reactjs.org/docs/refs-and-the-dom.html

https://blog.logrocket.com/how-to-use-react-createref-ea014ad09dba/

createRef & useRef

createRef

16.3引入了createRef這個api

React ref的原理和應用

createRef的源碼就是一個閉包,對外暴露了 一個具有 current屬性的對象。

我們一般會這樣在class component中使用createRef

class CreateRefComponent extends React.Component {   constructor(props) {     super(props);     this.myRef = React.createRef()   }   componentDidMount() {     this.myRef.current.focus()     console.log(this.myRef.current)     // dom input   }   render() {     return <input ref={this.myRef} />   } }

為什么不能在函數組件中使用createRef

結合第一節的內容以及  createRef的源碼,我們發現,這不過就是在類組件內部掛載了一個可變對象。因為類組件構造函數不會被反復執行,因此這個createRef自然保持同一份引用。但是到了函數組件就不一樣了,每一次組件更新,  因為沒有特殊處理createRef會被反復重新創建執行,因此在函數組件中使用createRef將不能達到只有同一份引用的效果。

const CreateRefInFC = () => {   const valRef = React.createRef();  // 如果在函數組件中使用createRef 在這個例子中點擊后ref就會被重新創建因此將始終顯示為null   const [, update] = React.useState();   return <div>     value: {valRef.current}     <button onClick={() => {       valRef.current = 80;       update({});     }}>+     </button>   </div> }

 useRef

React 16.8中出現了hooks,使得我們可以在函數組件中定義狀態,同時也帶來了 useRef

React ref的原理和應用

React ref的原理和應用

再來看moutRef和updateRef所做的事:

function mountRef(initialValue) {   var hook = mountWorkInProgressHook();   {     var _ref2 = {       current: initialValue     };     hook.memoizedState = _ref2;     return _ref2;   } } function updateRef(initialValue) {   var hook = updateWorkInProgressHook();   return hook.memoizedState; }

借助hook數據結構,第一次useRef時將創建的值保存在memoizedState中,之后每次更新階段則直接返回。

這樣在函數組件更新時重復執行useRef仍返回同一份引用。

因此實際上和 createRef一樣本質上只是創建了一個 Mutable  Object,只是因為渲染方式的不同,在函數組件中做了一些處理。而掛載和卸載的行為全部交由組件本身來維護。

被擴展的ref

從  createRef開始我們可以看到,ref對象的消費不再和DOM以及組件屬性所綁定了,這意味著你可以在任何地方消費他們,這也回答了本文一開始的那個問題。

useRef的應用

解決閉包問題

由于函數組件每次執行形成的閉包,下面這段代碼會始終打印1

export const ClosureDemo =  () => {     const [ count,setCount ] = useState(0);     useEffect(()=> {         const interval = setInterval(()=>{           setCount(count+1)         }, 1000)         return () => clearInterval(interval)       }, [])     // count顯示始終是1     return <div>{ count }</div> }

將 count 作為依賴傳入useEffect可以解決上面這個問題

export const ClosureDemo =  () => {     const [ count,setCount ] = useState(0);     useEffect(()=> {         const interval = setInterval(()=>{           setCount(count+1)         }, 1000)         return () => clearInterval(interval)       }, [count])     return <div>{ count }</div> }

但是這樣定時器也會隨著count值的更新而被不斷創建,一方面會帶來性能問題(這個例子中沒有那么明顯),更重要的一個方面是它不符合我們的開發語義,因為很明顯我們希望定時器本身是不變的。

另外一個方式也可以處理這個問題

export const ClosureDemo =  () => {     const [ count,setCount ] = useState(0);     useEffect(()=> {         const interval = setInterval(()=>{           setCount(count=> count + 1) // 使用setSate函數式更新可以確保每次都取到新的值         }, 1000)         return () => clearInterval(interval)       }, [])     return <div>{ count }</div> }

這樣做確實可以處理閉包帶來的影響,但是僅限于需要使用setState的場景,對數據的修改和觸發setState是需要綁定的,這可能會造成不必要的刷新。

使用useRef創建引用

export const ClosureDemo =  () => {     const [ count,setCount ] = useState(0);     const countRef = useRef(0);     countRef.current = count     useEffect(()=> {         const interval = setInterval(()=>{           // 這里將更新count的邏輯和觸發更新的邏輯解耦了           if(countRef.current < 5){             countRef.current++           } else {             setCount(countRef.current)           }         }, 1000)         return () => clearInterval(interval)       }, [])     return <div>{ count }</div> }

 封裝自定義hooks

useCreation

通過factory函數來避免類似于 useRef(new Construcotr)中構造函數的重復執行

import { useRef } from 'react'; export default function useCreation<T>(factory: () => T, deps: any[]) {   const { current } = useRef({     deps,     obj: undefined as undefined | T,     initialized: false,   });   if (current.initialized === false || !depsAreSame(current.deps, deps)) {     current.deps = deps;     current.obj = factory();     current.initialized = true;   }   return current.obj as T; } function depsAreSame(oldDeps: any[], deps: any[]): boolean {   if (oldDeps === deps) return true;   for (const i in oldDeps) {     if (oldDeps[i] !== deps[i]) return false;   }   return true; }

usePrevious

通過創建兩個ref來保存前一次的state

import { useRef } from 'react'; export type compareFunction<T> = (prev: T | undefined, next: T) => boolean; function usePrevious<T>(state: T, compare?: compareFunction<T>): T | undefined {   const prevRef = useRef<T>();   const curRef = useRef<T>();   const needUpdate = typeof compare === 'function' ? compare(curRef.current, state) : true;   if (needUpdate) {     prevRef.current = curRef.current;     curRef.current = state;   }   return prevRef.current; } export default usePrevious;

useClickAway

自定義的元素失焦響應hook

import { useEffect, useRef } from 'react'; export type BasicTarget<T = HTMLElement> =   | (() => T | null)   | T   | null   | MutableRefObject<T | null | undefined>;     export function getTargetElement(   target?: BasicTarget<TargetElement>,   defaultElement?: TargetElement, ): TargetElement | undefined | null {   if (!target) {     return defaultElement;   }   let targetElement: TargetElement | undefined | null;   if (typeof target === 'function') {     targetElement = target();   } else if ('current' in target) {     targetElement = target.current;   } else {     targetElement = target;   }   return targetElement; } // 鼠標點擊事件,click 不會監聽右鍵 const defaultEvent = 'click'; type EventType = MouseEvent | TouchEvent; export default function useClickAway(   onClickAway: (event: EventType) => void,   target: BasicTarget | BasicTarget[],   eventName: string = defaultEvent, ) {   // 使用useRef保存回調函數   const onClickAwayRef = useRef(onClickAway);   onClickAwayRef.current = onClickAway;   useEffect(() => {     const handler = (event: any) => {       const targets = Array.isArray(target) ? target : [target];       if (         targets.some((targetItem) => {           const targetElement = getTargetElement(targetItem) as HTMLElement;           return !targetElement || targetElement?.contains(event.target);         }) ) {         return;       }       onClickAwayRef.current(event);     };     document.addEventListener(eventName, handler);     return () => {       document.removeEventListener(eventName, handler);     };   }, [target, eventName]); }

以上自定義hooks均出自ahooks

還有許多好用的自定義hook以及倉庫比如react-use都基于useRef自定義了很多好用的hook。

“React ref的原理和應用”的內容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業相關的知識可以關注創新互聯網站,小編將為大家輸出更多高質量的實用文章!

網站名稱:Reactref的原理和應用
網站網址:http://m.newbst.com/article14/jhshde.html

成都網站建設公司_創新互聯,為您提供做網站自適應網站、靜態網站全網營銷推廣、外貿建站營銷型網站建設

廣告

聲明:本網站發布的內容(圖片、視頻和文字)以用戶投稿、用戶轉載內容為主,如果涉及侵權請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網站立場,如需處理請聯系客服。電話:028-86922220;郵箱:631063699@qq.com。內容未經允許不得轉載,或轉載時需注明來源: 創新互聯

小程序開發