Biarkan anda memahami Ref dalam React dan berkongsi mata pengetahuan yang patut diketahui.

青灯夜游
Lepaskan: 2022-03-22 11:21:34
ke hadapan
4460 orang telah melayarinya

Artikel ini akan membawa anda memahami Ref dalam React dan memperkenalkan beberapa perkara pengetahuan yang perlu anda ketahui tentang Ref. Saya harap ia akan membantu semua orang!

Biarkan anda memahami Ref dalam React dan berkongsi mata pengetahuan yang patut diketahui.

Pengenalan

Dalam projek React, terdapat banyak senario di mana Ref diperlukan. Contohnya, gunakan atribut ref untuk mendapatkan nod DOM dan dapatkan contoh objek ClassComponent; juga boleh memanggil kaedah useRef untuk menciptanya secara manual objek setInterval. [Cadangan berkaitan: React.createRefTutorial video RedisRef]Walaupun

sangat mudah digunakan, masih tidak dapat dielakkan untuk menghadapi masalah dalam projek sebenar Artikel ini akan bermula dari

Dari perspektif kod sumber Ref, kami menyelesaikan pelbagai isu berkaitan dan menjelaskan perkara yang dilakukan di sebalik API yang berkaitan dengan . Selepas membaca artikel ini, anda mungkin mempunyai pemahaman yang lebih mendalam tentang Ref. refRefRujuk pengisytiharan jenis berkaitan

Pertama sekali,

ialah singkatan daripada

, yang merupakan rujukan. Dalam fail pengisytiharan jenis ref, anda boleh menemui beberapa jenis yang berkaitan dengan Ruj, dan ia disenaraikan di sini. referencereactRefObject/MutableRefObject

Apabila menggunakan
interface RefObject<T> { readonly current: T | null; }
interface MutableRefObject<T> { current: T; }
Salin selepas log masuk
Hook, ia mengembalikan RefObject/MutableRefObejct kedua-dua jenis mentakrifkan struktur objek

> ialah useRefbaca sahaja{ current: T } Jika RefObject diubah suai, Skrip taip akan memberi amaran ⚠️. refObject.current

Ralat TS: Tidak boleh ditetapkan kepada "semasa" kerana ia adalah sifat baca sahaja.
const ref = useRef<string>(null)
ref.current = &#39;&#39; // Error
Salin selepas log masuk

Lihat takrif kaedah Biarkan anda memahami Ref dalam React dan berkongsi mata pengetahuan yang patut diketahui. Di sini,

lebih muatan

digunakan apabila parameter generik masuk useRef adalah tidak Mengembalikan apabila mengandungi dan T apabila mengandungi null. RefObject<T>nullMutableRefObject<T>Jadi jika anda mahu sifat semasa objek ref yang dicipta boleh diubah suai, anda perlu menambah

.
function useRef<T>(initialValue: T): MutableRefObject<T>;
function useRef<T>(initialValue: T | null): RefObject<T>;
Salin selepas log masuk

| nullApabila memanggil kaedah

, kaedah itu juga mengembalikan
const ref = useRef<string | null>(null)
ref.current = &#39;&#39; // OK
Salin selepas log masuk
.

React.createRef()RefObjectcreateRef

telah ditambahkan dalam versi
export function createRef(): RefObject {
  const refObject = {
    current: null,
  };
  if (__DEV__) {
    Object.seal(refObject);
  }
  return refObject;
}
Salin selepas log masuk
Jika anda menggunakan versi terdahulu, anda perlu menggunakan

. RefObject/MutableRefObject16.3Ref CallbackRefCallback

Menggunakan adalah untuk menghantar fungsi panggil balik Apabila bertindak balas panggilan balik, kejadian yang sepadan akan dihantar semula dan anda boleh menyimpannya sendiri memanggil. Jenis fungsi panggil balik ini ialah

.

Ref CallbackRefCallbackMenggunakan

Contoh:
type RefCallback<T> = (instance: T | null) => void;
Salin selepas log masuk

RefCallback

Ref/LegacyRef
import React from &#39;react&#39;

export class CustomTextInput extends React.Component {
  textInput: HTMLInputElement | null = null;

  saveInputRef = (element: HTMLInputElement | null) => {
    this.textInput = element;
  }

  render() {
    return (
      <input type="text" ref={this.saveInputRef} />
    );
  }
}
Salin selepas log masuk

Dalam pengisytiharan jenis, terdapat juga Ruj/ Jenis LegacyRef , ia digunakan untuk merujuk kepada jenis Ref secara amnya. ialah versi yang serasi Dalam versi lama sebelumnya,

juga boleh menjadi

rentetanLegacyRef. refHanya apabila anda memahami jenis yang berkaitan dengan Ref anda boleh menjadi lebih selesa menulis TypeScript.

type Ref<T> = RefCallback<T> | RefObject<T> | null;
type LegacyRef<T> = string | Ref<T>;
Salin selepas log masuk
Rujuk lulus

Alat khas

Apabila menggunakan pada komponen JSX, kami menetapkan

. Kita semua tahu bahawa sintaks

akan disusun ke dalam bentuk ref oleh alat seperti Babel. refRefjsxNampaknya createElement tidak berbeza dengan prop lain, tetapi jika anda cuba mencetak props.ref di dalam komponen itu adalah

. Dan
// jsx
<App ref={ref} id="my-app" ></App>

// compiled to
React.createElement(App, {
  ref: ref,
  id: "my-app"
});
Salin selepas log masuk
konsol persekitaran akan memberikan gesaan.

refundefinedCuba mengaksesnya akan menyebabkan dev dikembalikan Jika anda perlu mengakses nilai yang sama dalam komponen anak, anda harus menghantarnya sebagai prop yang berbeza.

React 对 ref 做了啥?在 ReactElement 源码中可以看到,refRESERVED_PROPS,同样有这种待遇的还有 key,它们都会被特殊处理,从 props 中提取出来传递给 Element

const RESERVED_PROPS = {
  key: true,
  ref: true,
  __self: true,
  __source: true,
};
Salin selepas log masuk

所以 ref 是会被特殊处理的 “props“

forwardRef

16.8.0 版本之前,Function Component 是无状态的,只会根据传入的 props render。有了 Hook 之后不仅可以有内部状态,还可以暴露方法供外部调用(需要借助 forwardRefuseImperativeHandle)。

如果直接对一个 Function Componentref,dev 环境下控制台会告警,提示你需要用 forwardRef 进行包裹起来。

function Input () {
    return <input />
}

const ref = useRef()
<Input ref={ref} />
Salin selepas log masuk

Function components cannot be given refs. Attempts to access this ref will fail. Did you mean to use React.forwardRef()?

forwardRef 为何物?查看源码 ReactForwardRef.js__DEV__ 相关的代码折叠起来,它只是一个无比简单的高阶组件。接收一个 render 的 FunctionComponent,将它包裹一下定义 $$typeofREACT_FORWARD_REF_TYPEreturn 回去。

Biarkan anda memahami Ref dalam React dan berkongsi mata pengetahuan yang patut diketahui.

跟踪代码,找到 resolveLazyComponentTag,在这里 $$typeof 会被解析成对应的 WorkTag。

Biarkan anda memahami Ref dalam React dan berkongsi mata pengetahuan yang patut diketahui.

REACT_FORWARD_REF_TYPE 对应的 WorkTag 是 ForwardRef。紧接着 ForwardRef 又会进入 updateForwardRef 的逻辑。

case ForwardRef: {
  child = updateForwardRef(
    null,
    workInProgress,
    Component,
    resolvedProps,
    renderLanes,
  );
  return child;
}
Salin selepas log masuk

这个方法又会调用 renderWithHooks 方法,并在第五个参数传入 ref

nextChildren = renderWithHooks(
  current,
  workInProgress,
  render,
  nextProps,
  ref, // 这里
  renderLanes,
);
Salin selepas log masuk

继续跟踪代码,进入 renderWithHooks 方法,可以看到,ref 会作为 Component 的第二个参数传递。到这里我们可以理解被 forwardRef 包裹的 FuncitonComponent 第二个参数 ref 是从哪里来的(对比 ClassComponent contructor 第二个参数是 Context)。

Biarkan anda memahami Ref dalam React dan berkongsi mata pengetahuan yang patut diketahui.

了解如何传递 ref,那下一个问题就是 ref 是如何被赋值的。

ref 的赋值

打断点(给 ref 赋值一个 RefCallback,在 callback 里面打断点) 跟踪到代码 commitAttachRef,在这个方法里面,会判断 Fiber 节点的 ref 是 function 还是 RefObject,依据类型处理 instance。如果这个 Fiber 节点是 HostComponent (tag = 5) 也就是 DOM 节点,instance 就是该 DOM 节点;而如果该 Fiber 节点是 ClassComponent (tag = 1),instance 就是该对象实例。

function commitAttachRef(finishedWork) {
  var ref = finishedWork.ref;

  if (ref !== null) {
    var instanceToUse = finishedWork.stateNode;

    if (typeof ref === &#39;function&#39;) {
      ref(instanceToUse);
    } else {
      ref.current = instanceToUse;
    }
  }
}
Salin selepas log masuk

以上是 HostComponent 和 ClassComponent 中对 ref 的赋值逻辑,对于 ForwardRef 类型的组件走的是另外的代码,但行为基本是一致的,可以看这里 imperativeHandleEffect

接下里,我们继续挖掘 React 源码,看看 useRef 是如何实现的。

useRef 的内部实现

通过跟踪代码,定位到 useRef 运行时的代码 ReactFiberHooks

Biarkan anda memahami Ref dalam React dan berkongsi mata pengetahuan yang patut diketahui.

这里有两个方法,mountRefupdateRef,顾名思义就是对应 Fiber 节点 mountupdate 时对 ref 的操作。

function updateRef<T>(initialValue: T): {|current: T|} {
  const hook = updateWorkInProgressHook();
  return hook.memoizedState;
}

function mountRef<T>(initialValue: T): {|current: T|} {
  const hook = mountWorkInProgressHook();
  const ref = {current: initialValue};
  hook.memoizedState = ref;
  return ref;
}
Salin selepas log masuk

可以看到 mount 时,useRef 创建了一个 RefObject,并将它赋值给 hookmemoizedStateupdate 时直接将它取出返回。

不同的 Hook memoizedState 保存的内容不一样,useState 中保存 state 信息, useEffect 中 保存着 effect 对象,useRef 中保存的是 ref 对象...

mountWorkInProgressHookupdateWorkInProgressHook 方法背后是一条 Hooks 的链表,在不修改链表的情况下,每次 render useRef 都能取回同一个 memoizedState 对象,就这么简单。

应用:合并 ref

至此,我们了解了在 React 中 ref 的传递和赋值逻辑,以及 useRef 相关的源码。用一个应用题来巩固以上知识点:有一个 Input 组件,在组件内部需要通过 innerRef HTMLInputElement 来访问 DOM 节点,同时也允许组件外部 ref 该节点,需要怎么实现?

const Input = forwardRef((props, ref) => {
  const innerRef = useRef<HTMLInputElement>(null)
  return (
    <input {...props} ref={???} />
  )
})
Salin selepas log masuk

考虑一下上面代码中的 ??? 应该怎么写。

============ 答案分割线 ==============

通过了解 Ref 相关的内部实现,很明显我们这里可以创建一个 RefCallback,在里面对多个 ref 进行赋值就可以了。

export function combineRefs<T = any>(
  refs: Array<MutableRefObject<T | null> | RefCallback<T>>
): React.RefCallback<T> {
  return value => {
    refs.forEach(ref => {
      if (typeof ref === &#39;function&#39;) {
        ref(value);
      } else if (ref !== null) {
        ref.current = value;
      }
    });
  };
}

const Input = forwardRef((props, ref) => {
  const innerRef = useRef<HTMLInputElement>(null)
  return (
    <input {...props} ref={combineRefs(ref, innerRef)} />
  )
})
Salin selepas log masuk

更多编程相关知识,请访问:编程入门!!

Atas ialah kandungan terperinci Biarkan anda memahami Ref dalam React dan berkongsi mata pengetahuan yang patut diketahui.. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!

Label berkaitan:
sumber:juejin.cn
Kenyataan Laman Web ini
Kandungan artikel ini disumbangkan secara sukarela oleh netizen, dan hak cipta adalah milik pengarang asal. Laman web ini tidak memikul tanggungjawab undang-undang yang sepadan. Jika anda menemui sebarang kandungan yang disyaki plagiarisme atau pelanggaran, sila hubungi admin@php.cn
Tutorial Popular
Lagi>
Muat turun terkini
Lagi>
kesan web
Kod sumber laman web
Bahan laman web
Templat hujung hadapan
Tentang kita Penafian Sitemap
Laman web PHP Cina:Latihan PHP dalam talian kebajikan awam,Bantu pelajar PHP berkembang dengan cepat!