


Lassen Sie Ref in React verstehen und wissenswerte Wissenspunkte teilen.
Dieser Artikel wird Ihnen helfen, Ref in React zu verstehen und einige Wissenspunkte vorzustellen, die Sie über Ref wissen müssen. Ich hoffe, dass er für alle hilfreich ist!
Einführung
In React-Projekten gibt es viele Szenarien, in denen Ref
benötigt wird. Verwenden Sie beispielsweise das Attribut ref
, um den DOM-Knoten und die Instanz des ClassComponent-Objekts abzurufen. Verwenden Sie den Hook useRef
, um ein Ref-Objekt zu erstellen, um Probleme wie setInterval zu lösen
nicht in der Lage ist, das neueste Statusproblem abzurufen; Sie können auch die Methode React.createRef
aufrufen, um manuell ein Ref
-Objekt zu erstellen. [Verwandte Empfehlungen: Redis-Video-Tutorial]Ref
。例如使用 ref
属性获取 DOM 节点,获取 ClassComponent 对象实例;用 useRef
Hook 创建一个 Ref 对象,以便解决像 setInterval
获取不到最新的 state 的问题;你也可以调用 React.createRef
方法手动创建一个 Ref
对象。【相关推荐:Redis视频教程】
虽然 Ref
用起来也很简单,但在实际项目中实战还是难免遇到问题,这篇文章将从源码的角度出发梳理各种和 Ref
相关的问题,理清和 ref
相关的 API 背后都干了什么。看完这篇文章或许可以让你对的 Ref
有更深入地认识。
Ref 相关的类型声明
首先 ref
是 reference
的简称,也就是引用。在 react
的类型声明文件中,可以找到好几个和 Ref 相关的类型,这里将它们一一列举出来。
RefObject/MutableRefObject
interface RefObject<T> { readonly current: T | null; } interface MutableRefObject<T> { current: T; }
使用 useRef
Hook 的时候返回的就是 RefObject/MutableRefObejct,这两个类型都是定义了一个 { current: T }
的对象结构,区别是 RefObject
的 current 属性是只读的,如果修改 refObject.current
,Typescript 会警告⚠️。
const ref = useRef<string>(null) ref.current = '' // Error
TS 报错:无法分配到 "current" ,因为它是只读属性。
查看 useRef
方法的定义,这里用了函数重载,当传入的泛型参数 T
不包含 null
时返回RefObject<T>
,当包含 null
时将返回 MutableRefObject<T>
。
function useRef<T>(initialValue: T): MutableRefObject<T>; function useRef<T>(initialValue: T | null): RefObject<T>;
所以如果你希望创建的 ref 对象 current 属性是可修改的,需要加上 | null
。
const ref = useRef<string | null>(null) ref.current = '' // OK
调用 React.createRef()
方法时返回的也是一个 RefObject
。
export function createRef(): RefObject { const refObject = { current: null, }; if (__DEV__) { Object.seal(refObject); } return refObject; }
RefObject/MutableRefObject
是在 16.3
版本才新增的,如果使用更早的版本,需要使用 Ref Callback
。
RefCallback
使用 Ref Callback
就是传递一个回调函数,react 回调时会将对应的实例回传过来,可以自行保存以便调用。这个回调函数的类型就是 RefCallback
。
type RefCallback<T> = (instance: T | null) => void;
使用 RefCallback
示例:
import React from 'react' 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} /> ); } }
Ref/LegacyRef
在类型声明中,还有 Ref/LegacyRef 类型,它们用于泛指 Ref 类型。 LegacyRef
是兼容版本,在之前的老版本 ref
还可以是 字符串。
type Ref<T> = RefCallback<T> | RefObject<T> | null; type LegacyRef<T> = string | Ref<T>;
理解了和 Ref 相关的类型,写起 Typescript 来才能更得心应手。
Ref 的传递
特殊的 props
在 JSX 组件上使用 ref
时,我们是通过给 ref
属性设置一个 Ref
。我们都知道 jsx
的语法,会被 Babel 等工具编译成 createElement
的形式。
// jsx <App ref={ref} id="my-app" ></App> // compiled to React.createElement(App, { ref: ref, id: "my-app" });
看起来 ref
和其他 prop 没啥区别,不过如果你尝试在组件内部打印 props.ref 却是 undefined
。并且 dev
环境控制台会给出提示。
Trying to access it will result in
Obwohlundefined
Ref</ code> ist ebenfalls sehr einfach zu verwenden, aber es ist unvermeidlich, dass in tatsächlichen Projekten Probleme auftreten. In diesem Artikel werden verschiedene Probleme im Zusammenhang mit <code>Ref
aus der Quellcode-Perspektive geklärt , klären Sie, was hinter der API im Zusammenhang mitref
geschieht. Nachdem Sie diesen Artikel gelesen haben, haben Sie möglicherweise ein tieferes Verständnis vonRef
. 🎜Ref-bezogene Typdeklaration🎜🎜Zuallererst ist
ref
die Abkürzung fürreference
, also eine Referenz. In der Typdeklarationsdatei vonreact
finden Sie mehrere Typen, die sich auf Ref beziehen. Sie sind hier aufgelistet. 🎜RefObject/MutableRefObject
🎜Bei Verwendung desconst RESERVED_PROPS = { key: true, ref: true, __self: true, __source: true, };Nach dem Login kopierenNach dem Login kopierenuseRef
-Hooks wird RefObject/MutableRefObejct zurückgegeben. Die Objektstruktur von < code>{ current: T , der Unterschied besteht darin, dass die aktuelle Eigenschaft vonRefObject
schreibgeschützt ist, wenn SierefObject.current< ändern /code>, Typescript gibt eine Warnung aus ⚠️. 🎜<div class="code" style="position:relative; padding:0px; margin:0px;"><div class="code" style="position:relative; padding:0px; margin:0px;"><pre class='brush:php;toolbar:false;'>function Input () { return <input /> } const ref = useRef() <Input ref={ref} /></pre><div class="contentsignin">Nach dem Login kopieren</div></div><div class="contentsignin">Nach dem Login kopieren</div></div>🎜<small>TS-Fehler: Kann „current“ nicht zugewiesen werden, da es sich um eine schreibgeschützte Eigenschaft handelt. </small>🎜🎜<img src="/static/imghw/default1.png" data-src="https://img.php.cn/upload/image/187/445/797/1647918576134359.png" class="lazy" title="164791853547070Lassen Sie Ref in React verstehen und wissenswerte Wissenspunkte teilen." alt="Lassen Sie Ref in React verstehen und wissenswerte Wissenspunkte teilen."/ >🎜🎜Sehen Sie sich die Definition der <code>useRef
-Methode an, die hier verwendet wird, wenn der eingehende generische ParameterT
keinenenthält
enthalten ist, undRefObject<T>
wird zurückgegeben, wenn nullMutableRefObject<T>
wird zurückgegeben, wennnull
enthalten ist. 🎜🎜Wenn Sie also möchten, dass die aktuelle Eigenschaft des erstellten Referenzobjekts geändert werden kann, müssen Siecase ForwardRef: { child = updateForwardRef( null, workInProgress, Component, resolvedProps, renderLanes, ); return child; }Nach dem Login kopierenNach dem Login kopieren| null
hinzufügen. 🎜🎜Beim Aufruf der MethodenextChildren = renderWithHooks( current, workInProgress, render, nextProps, ref, // 这里 renderLanes, );Nach dem Login kopierenNach dem Login kopierenReact.createRef()
wird auch einRefObject
zurückgegeben. 🎜🎜createRef🎜🎜function commitAttachRef(finishedWork) { var ref = finishedWork.ref; if (ref !== null) { var instanceToUse = finishedWork.stateNode; if (typeof ref === 'function') { ref(instanceToUse); } else { ref.current = instanceToUse; } } }Nach dem Login kopierenNach dem Login kopierenRefObject/MutableRefObject
wurde in der Version16.3
hinzugefügt. Wenn Sie eine frühere Version verwenden, müssen Sieverwenden Ref-Rückruf
. 🎜RefCallback
🎜MitRef Callback
wird eine Rückruffunktion übergeben und React ruft die entsprechende Instanz auf beim Rückruf Wenn es zurückgegeben wird, kann es für einen einfachen Anruf gespeichert werden. Der Typ dieser Rückruffunktion istRefCallback
. 🎜🎜Verwenden Siefunction 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; }Nach dem Login kopierenNach dem Login kopierenRefCallback
Beispiel: 🎜const Input = forwardRef((props, ref) => { const innerRef = useRef<HTMLInputElement>(null) return ( <input {...props} ref={???} /> ) })Nach dem Login kopierenNach dem Login kopierenRef/LegacyRef
🎜In der Typdeklaration und Ref /LegacyRef-Typen werden verwendet, um allgemein auf den Ref-Typ zu verweisen.LegacyRef
ist eine kompatible Version. In der vorherigen alten Version kannref
auch string. 🎜🎜Nur wenn Sie die mit Ref verbundenen Typen verstehen, können Sie sich beim Schreiben von Typescript wohler fühlen. 🎜export function combineRefs<T = any>( refs: Array<MutableRefObject<T | null> | RefCallback<T>> ): React.RefCallback<T> { return value => { refs.forEach(ref => { if (typeof ref === 'function') { 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)} /> ) })Nach dem Login kopierenNach dem Login kopierenÜbergabe von Ref 🎜
Spezielle Requisiten
🎜Wird für JSX-Komponenten verwendet< Code> ref setzen wir einenRef
auf das Attributref
. Wir alle wissen, dass die Syntax vonjsx
von Tools wie Babel in die Form voncreateElement
kompiliert wird. 🎜rrreee🎜Es scheint, dassref
sich nicht von anderen Requisiten unterscheidet, aber wenn Sie versuchen, props.ref innerhalb der Komponente zu drucken, wird esundefiniert
sein. Und diedev
-Umgebungskonsole gibt eine Eingabeaufforderung aus. 🎜🎜Der Versuch, darauf zuzugreifen, führt dazu, dassundefiniert
zurückgegeben wird. Wenn Sie auf denselben Wert innerhalb der untergeordneten Komponente zugreifen müssen, sollten Sie ihn als andere Requisite übergeben.🎜React 对 ref 做了啥?在 ReactElement 源码中可以看到,
ref
是RESERVED_PROPS
,同样有这种待遇的还有key
,它们都会被特殊处理,从 props 中提取出来传递给Element
。const RESERVED_PROPS = { key: true, ref: true, __self: true, __source: true, };Nach dem Login kopierenNach dem Login kopieren所以
ref
是会被特殊处理的“props“
。forwardRef
在
16.8.0
版本之前,Function Component 是无状态的,只会根据传入的 props render。有了 Hook 之后不仅可以有内部状态,还可以暴露方法供外部调用(需要借助forwardRef
和useImperativeHandle
)。如果直接对一个
Function Component
用ref
,dev 环境下控制台会告警,提示你需要用forwardRef
进行包裹起来。function Input () { return <input /> } const ref = useRef() <Input ref={ref} />Nach dem Login kopierenNach dem Login kopierenFunction 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,将它包裹一下定义$$typeof
为REACT_FORWARD_REF_TYPE
,return
回去。
跟踪代码,找到 resolveLazyComponentTag,在这里
$$typeof
会被解析成对应的 WorkTag。
REACT_FORWARD_REF_TYPE
对应的 WorkTag 是 ForwardRef。紧接着 ForwardRef 又会进入 updateForwardRef 的逻辑。case ForwardRef: { child = updateForwardRef( null, workInProgress, Component, resolvedProps, renderLanes, ); return child; }Nach dem Login kopierenNach dem Login kopieren这个方法又会调用 renderWithHooks 方法,并在第五个参数传入
ref
。nextChildren = renderWithHooks( current, workInProgress, render, nextProps, ref, // 这里 renderLanes, );Nach dem Login kopierenNach dem Login kopieren继续跟踪代码,进入 renderWithHooks 方法,可以看到,
ref
会作为Component
的第二个参数传递。到这里我们可以理解被forwardRef
包裹的FuncitonComponent
第二个参数ref
是从哪里来的(对比 ClassComponent contructor 第二个参数是 Context)。
了解如何传递 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 === 'function') { ref(instanceToUse); } else { ref.current = instanceToUse; } } }Nach dem Login kopierenNach dem Login kopieren以上是 HostComponent 和 ClassComponent 中对 ref 的赋值逻辑,对于 ForwardRef 类型的组件走的是另外的代码,但行为基本是一致的,可以看这里 imperativeHandleEffect。
接下里,我们继续挖掘 React 源码,看看 useRef 是如何实现的。
useRef 的内部实现
通过跟踪代码,定位到 useRef 运行时的代码 ReactFiberHooks
这里有两个方法,
mountRef
和updateRef
,顾名思义就是对应Fiber
节点mount
和update
时对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; }Nach dem Login kopierenNach dem Login kopieren可以看到
mount
时,useRef
创建了一个RefObject
,并将它赋值给hook
的memoizedState
,update
时直接将它取出返回。不同的 Hook memoizedState 保存的内容不一样,
useState
中保存state
信息,useEffect
中 保存着effect
对象,useRef
中保存的是ref
对象...
mountWorkInProgressHook
,updateWorkInProgressHook
方法背后是一条 Hooks 的链表,在不修改链表的情况下,每次 render useRef 都能取回同一个 memoizedState 对象,就这么简单。应用:合并 ref
至此,我们了解了在 React 中
ref
的传递和赋值逻辑,以及useRef
相关的源码。用一个应用题来巩固以上知识点:有一个 Input 组件,在组件内部需要通过 innerRefHTMLInputElement
来访问DOM
节点,同时也允许组件外部 ref 该节点,需要怎么实现?const Input = forwardRef((props, ref) => { const innerRef = useRef<HTMLInputElement>(null) return ( <input {...props} ref={???} /> ) })Nach dem Login kopierenNach dem Login kopieren考虑一下上面代码中的
???
应该怎么写。============ 答案分割线 ==============
通过了解 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 === 'function') { 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)} /> ) })Nach dem Login kopierenNach dem Login kopieren更多编程相关知识,请访问:编程入门!!
Das obige ist der detaillierte Inhalt vonLassen Sie Ref in React verstehen und wissenswerte Wissenspunkte teilen.. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Heiße KI -Werkzeuge

Undresser.AI Undress
KI-gestützte App zum Erstellen realistischer Aktfotos

AI Clothes Remover
Online-KI-Tool zum Entfernen von Kleidung aus Fotos.

Undress AI Tool
Ausziehbilder kostenlos

Clothoff.io
KI-Kleiderentferner

AI Hentai Generator
Erstellen Sie kostenlos Ai Hentai.

Heißer Artikel

Heiße Werkzeuge

Notepad++7.3.1
Einfach zu bedienender und kostenloser Code-Editor

SublimeText3 chinesische Version
Chinesische Version, sehr einfach zu bedienen

Senden Sie Studio 13.0.1
Leistungsstarke integrierte PHP-Entwicklungsumgebung

Dreamweaver CS6
Visuelle Webentwicklungstools

SublimeText3 Mac-Version
Codebearbeitungssoftware auf Gottesniveau (SublimeText3)

Heiße Themen



So erstellen Sie eine Echtzeit-Chat-Anwendung mit React und WebSocket Einführung: Mit der rasanten Entwicklung des Internets hat die Echtzeitkommunikation immer mehr Aufmerksamkeit auf sich gezogen. Live-Chat-Apps sind aus dem modernen Sozial- und Arbeitsleben nicht mehr wegzudenken. In diesem Artikel wird erläutert, wie Sie mit React und WebSocket eine einfache Echtzeit-Chat-Anwendung erstellen, und es werden spezifische Codebeispiele bereitgestellt. 1. Technische Vorbereitung Bevor wir mit der Erstellung einer Echtzeit-Chat-Anwendung beginnen, müssen wir die folgenden Technologien und Tools vorbereiten: React: eine zum Erstellen

React-Leitfaden zur Front-End- und Back-End-Trennung: So erreichen Sie die Front-End- und Back-End-Entkopplung und die unabhängige Bereitstellung. Es sind spezifische Codebeispiele erforderlich. In der heutigen Webentwicklungsumgebung ist die Front-End- und Back-End-Trennung zu einem Trend geworden. Durch die Trennung von Front-End- und Back-End-Code kann die Entwicklungsarbeit flexibler und effizienter gestaltet und die Zusammenarbeit im Team erleichtert werden. In diesem Artikel wird erläutert, wie Sie mithilfe von React eine Front-End- und Back-End-Trennung erreichen und so die Ziele der Entkopplung und unabhängigen Bereitstellung erreichen. Zuerst müssen wir verstehen, was Front-End- und Back-End-Trennung ist. Im traditionellen Webentwicklungsmodell sind Front-End und Back-End gekoppelt

So erstellen Sie mit React und Flask einfache und benutzerfreundliche Webanwendungen. Einführung: Mit der Entwicklung des Internets werden die Anforderungen an Webanwendungen immer vielfältiger und komplexer. Um den Anforderungen der Benutzer an Benutzerfreundlichkeit und Leistung gerecht zu werden, wird es immer wichtiger, moderne Technologie-Stacks zum Aufbau von Netzwerkanwendungen zu verwenden. React und Flask sind zwei sehr beliebte Frameworks für die Front-End- und Back-End-Entwicklung, und sie arbeiten gut zusammen, um einfache und benutzerfreundliche Webanwendungen zu erstellen. In diesem Artikel erfahren Sie, wie Sie React und Flask nutzen

So erstellen Sie eine zuverlässige Messaging-Anwendung mit React und RabbitMQ Einführung: Moderne Anwendungen müssen zuverlässiges Messaging unterstützen, um Funktionen wie Echtzeitaktualisierungen und Datensynchronisierung zu erreichen. React ist eine beliebte JavaScript-Bibliothek zum Erstellen von Benutzeroberflächen, während RabbitMQ eine zuverlässige Messaging-Middleware ist. In diesem Artikel wird erläutert, wie Sie React und RabbitMQ kombinieren, um eine zuverlässige Messaging-Anwendung zu erstellen, und es werden spezifische Codebeispiele bereitgestellt. RabbitMQ-Übersicht:

React Responsive Design Guide: So erzielen Sie adaptive Front-End-Layouteffekte Mit der Beliebtheit mobiler Geräte und der steigenden Nachfrage der Benutzer nach Multi-Screen-Erlebnissen ist Responsive Design zu einem der wichtigsten Aspekte in der modernen Front-End-Entwicklung geworden. React, eines der derzeit beliebtesten Frontend-Frameworks, bietet eine Fülle von Tools und Komponenten, die Entwicklern dabei helfen, adaptive Layouteffekte zu erzielen. In diesem Artikel werden einige Richtlinien und Tipps zur Implementierung von responsivem Design mit React vorgestellt und spezifische Codebeispiele als Referenz bereitgestellt. Fle mit React

React-Code-Debugging-Leitfaden: So finden und beheben Sie Front-End-Fehler schnell. Einführung: Bei der Entwicklung von React-Anwendungen stoßen Sie häufig auf eine Vielzahl von Fehlern, die zum Absturz der Anwendung oder zu fehlerhaftem Verhalten führen können. Daher ist die Beherrschung von Debugging-Fähigkeiten eine wesentliche Fähigkeit für jeden React-Entwickler. In diesem Artikel werden einige praktische Techniken zum Auffinden und Beheben von Front-End-Fehlern vorgestellt und spezifische Codebeispiele bereitgestellt, um Lesern dabei zu helfen, Fehler in React-Anwendungen schnell zu finden und zu beheben. 1. Auswahl der Debugging-Tools: In Re

ReactRouter-Benutzerhandbuch: So implementieren Sie die Front-End-Routing-Steuerung Mit der Popularität von Single-Page-Anwendungen ist das Front-End-Routing zu einem wichtigen Bestandteil geworden, der nicht ignoriert werden kann. Als beliebteste Routing-Bibliothek im React-Ökosystem bietet ReactRouter umfangreiche Funktionen und benutzerfreundliche APIs, wodurch die Implementierung des Front-End-Routings sehr einfach und flexibel ist. In diesem Artikel wird die Verwendung von ReactRouter vorgestellt und einige spezifische Codebeispiele bereitgestellt. Um ReactRouter zuerst zu installieren, benötigen wir

So verwenden Sie React und Google BigQuery zum Erstellen schneller Datenanalyseanwendungen. Einführung: Im heutigen Zeitalter der Informationsexplosion ist die Datenanalyse zu einem unverzichtbaren Bindeglied in verschiedenen Branchen geworden. Unter anderem ist die Entwicklung schneller und effizienter Datenanalyseanwendungen für viele Unternehmen und Einzelpersonen zum Ziel geworden. In diesem Artikel wird erläutert, wie Sie mit React und Google BigQuery eine schnelle Datenanalyseanwendung erstellen, und es werden detaillierte Codebeispiele bereitgestellt. 1. Übersicht React ist ein Werkzeug zum Erstellen
