React 中的 forwardRef 究竟该怎么用?
React 中的 forwardRef
前言
最近在学react
, 一直对forwardRef
和Context
这两个认识不太清楚,现在终于搞懂了,于是想写一篇文章记录一下。
React.forwardRef
对于ref转发,官网是这样描述的
Ref 转发是一项将 ref 自动地通过组件传递到其一子组件的技巧。对于大多数应用中的组件来说,这通常不是必需的。但其对某些组件,尤其是可重用的组件库是很有用的。
想详细了解的可以去看官方文档 Refs 转发 – React (docschina.org)
现在让我们直奔主题吧!
React.forwardRef(render)
的返回值是react
组件,接收的参数是一个 render
函数,函数签名为render(props, ref)
,第二个参数将其接受的 ref 属性转发到render
返回的组件中。
这项技术并不常见,但在以下两种场景中特别有用:
- 转发
ref
到组件内部的DOM
节点上 - 在高阶组件中转发
ref
转发 ref
到组件内部的DOM
节点
比如我们想要将一个组件内部的某个元素暴露出去, 就可以这么做
// App.js
import React from 'react';
import Foo from './component/Foo';
export default class App extends React.Component {
constructor(props) {
super(props);
this.state = {};
this.input = React.createRef(); // 1
// ↑5
}
handleClick = (e) => {
const input = this.input.current;
// 6
console.log(input);
console.log(input.value);
input.focus();
}
render() {
return (
<>
<button onClick={this.handleClick}>click to get value</button>
{/*2*/}
<Foo ref={this.input}/>
</>
)
}
}
// Foo.jsx
import React from 'react';
// 3
const Foo = React.forwardRef((props, myRef) => {
return (
<div>
<p>....一些其他节点</p> {/*4*/}
<input type="text" defaultValue='ref 成功转发到 Foo 组件内部的 input节点上' ref={myRef}/>
<p>....一些其他节点</p>
<p>....一些其他节点</p>
</div>
);
});
export default Foo;
仔细看代码中标记的数字,这是ref
转发的流程:
- 创建了一个
ref
- 将其挂载到 组件上这个组件是通过
React.forwardRef
创建出来的, 注意这里很关键,后面细说 - 组件
Foo
接收到了一个ref
,于是将它转发到DOM
节点input
上 ref
如愿的挂载到内部节点input
上- 现在
this.input.current
保存着对节点input
的引用 - 点击按钮, 现在可以很轻松的获取
Foo
内部节点的value
以及获取其焦点
细节补充
之前说过,步骤2很关键,这是因为 ref
的值根据节点的类型而有所不同:
-
当
ref
属性用于 HTML 元素时,接收底层 DOM 元素作为其current
属性。 -
当
ref
属性用于自定义 class 组件时,ref
接收组件实例作为其current
属性。 -
不能在函数组件上使用
ref
属性,因为他们没有实例。
第一个很好理解,我们上面的例子已经体现了这一点, ref
最终被挂载到了 input
节点上, input
是一个HTML
元素,所以current
中保留的是DOM
元素。
对于第二个,我们现在用另一个组件来演示
// Bar.jsx
import React from 'react';
export default class Bar extends React.Component {
constructor(props) {
super(props);
this.state = {
message: '这是class组件, ref 只能挂载到实例上'
};
}
componentDidMount() {
console.log(this);
}
render() {
return (
<div>
class 组件
</div>
);
}
}
// App.js
import React from 'react';
import Bar from './component/Bar'
export default class App extends React.Component {
constructor(props) {
super(props);
this.state = {};
this.myRef = React.createRef(); // 创建 ref
}
handleClick = (e) => {
const instance = this.myRef.current;
// 打印的是 Bar 实例
console.log(instance);
}
render() {
return (
<div>
<button onClick={this.handleClick}>click to get instance</button>
{/*挂载到组件上,因为Bar是一个class组件,所以只能挂载到其实例上*/}
<Bar ref={this.myRef} />
</div>
);
}
}
第一条打印是 Bar 组件挂载后生命周期函数打印的
第二条打印是点击按钮后打印的,证明确实只是挂载到了组件实例上。后面的高阶组件中还会出现类似的问题。
对于第三个,不能在函数组件上使用 ref
属性,因为他们没有实例
// Baz.jsx
import React from 'react';
const Baz = (props) => {
return (
<div>
啊?
</div>
);
};
export default Baz;
// App.js 中, 省略掉了其他代码
<Baz ref={this.myRef} />
这时就会报错了, 意思是不能在函数式组件上使用 ref
, 尝试访问 ref
会失败。
一般函数式组件都是用React.forwardRef
包装一下然后返回出去的, 函数式组件本来就是一个render
函数,不过在被React.forwardRef
包装后就多了一个ref
属性了。
// 将我们的函数式组件改造成这个样子.
const Baz = React.forwardRef((props, ref) => {
return (
<div>
啊?
</div>
);
})
此时的ref
还没有被挂载,所以访问ref.current
会得到null
,不过总算不会报错了
函数式组件只是将ref
传递下去,ref
最终只能被挂载到内部的某个class组件或者HTML元素上
还有一点要说明一下,不能在函数组件上使用 ref
属性并不是不能在函数式组件内部使用 ref
如下所示:
// 还是 Baz.jsx
import React from 'react';
const myRef = React.createRef();
const Baz = (props) => {
function handleClick(e) {
const input = myRef.current;
console.log(input.value);
}
return (
<div>
<button onClick={handleClick}>click to get value</button>
<input ref={myRef} type="text" defaultValue={'不能在函数组件上使用 ref 属性并不代表着不能在函数式组件内部使用 `ref`'}/>
</div>
);
}
export default Baz;
在高阶组件中转发ref
依然使用 之前的 Bar.jsx
// Bar.jsx
import React from 'react';
export default class Bar extends React.Component {
constructor(props) {
super(props);
this.state = {
message: '这是Bar.jsx'
};
}
componentDidMount() {
console.log(this);
}
render() {
return (
<div>
class 组件
</div>
);
}
}
我们使用高阶组件,为Bar
组件增加一个功能:每次props
改变都打印其变化
// logProps.js
function logProps(WrappedComponent) {
class LogProps extends React.Component {
componentDidUpdate(prevProps) {
console.log('Previous props: ', prevProps);
console.log('Current props: ', this.props);
}
render() {
// 高阶组件透传所有的 props 到其包裹的组件,所以渲染结果将是相同的
return <WrappedComponent {...this.props} />;
}
}
return LogProps;
}
但是高阶组件 不! 会! 传! 递! ref
, 这是因为 ref
不是 prop
属性。就像 key
一样,其被 React 进行了特殊处理。
如果你对被高阶组件包装后的组件添加 ref
,该 ref
将引用最外层的容器组件,而不是被包裹的组件。
对于上面的例子,如果用了ref
, 那么最终会挂载到 <LogProps/>
组件上,而不是传入的被包裹的 <WrappedComponent/>
组件上。
其实这很好理解,自己在脑袋里模拟一下数据流就知道 ref 最后会被挂载到最外面的组件上,不过高阶组件中的这个透传的概念很容易将人带偏,误以为 ref 会跟着 props 一起透传下去, 其实是不会传递的。
为了更好的在控制台观察究竟是挂载到哪个组件上,我们为这两个组件添加 state
// Bar.jsx 中
this.state = {
message: '这是Bar.jsx'
}
// logProps.js 中返回的 LogProps组件
this.state = {
message: '这是LogProps'
}
// App.js
import React from 'react';
import logProps from './component/logProps';
import Bar from './component/Bar'
const BarWithLogProps = logProps(Bar);
export default class App extends React.Component {
constructor(props) {
super(props);
this.state = {
name: 'Roman',
age: 23,
hobby: 'video game'
}
this.myRef = React.createRef();
}
handleClick = (e) => {
this.setState({
name: 'fxy',
age: 32,
hobby: 'swim'
});
console.log(this.myRef.current);
}
render() {
return (
<div>
<button onClick={this.handleClick} >click to change props</button>
<BarWithLogProps {...this.state} ref={this.myRef} />
</div>
);
}
}
我们点击按钮,在控制台可以清楚的看到, ref
确实是被挂载到外部组件 LogProps
上。
React.forwardRef
再次登场,我们可以使用 React.forwardRef
API 明确地将 ref
转发到内部的 <WrappedComponent/>
组件
最后我们将 logProps.js
改造成这样
// 最终形态
import React from 'react';
export default function logProps(WrappedComponent) {
class LogProps extends React.Component {
// 2
constructor(props) {
super(props);
this.state = {
message: '这是LogProps'
}
}
componentDidUpdate(prevProps) {
console.log('Previous props: ', prevProps);
console.log('Current props: ', this.props);
}
render() {
// 3
const {customRef, ...props} = this.props;
// 3.5 return <WrappedComponent {...this.props}/>;
return <WrappedComponent {...props} ref={customRef} />;
}
}
// return LogProps;
return React.forwardRef((props, ref) => (
// 1
<LogProps {...props} customRef={ref} />
))
}
- 我们将
LogProps
组件作为render
函数的返回值,这样渲染结果还是不变,然后将传入的ref
转发到LogProps
组件的自定义属性customRef
上。注意这里一定要转发到自定义属性,如果转发到ref
属性上最终还是会被挂载到LogProps
上,等于转了一圈又回到了原地... - 所有的属性都被传递到
props
中 - 将
props
中的customRef
提取出来,最终传递到WrappedComponent
的ref
属性上。
最后ref
被成功转发到被包裹的组件WrappedComponent
上。
点击按钮,现在来控制台看看
ref 被成功的挂载到目标组件上。
细心的读者可能已经发现logProps.js
转发ref
还有另一种写法, 就是 3.5 那样,不用抽离 customRef
,仍然将 customRef
作为props
透传下去, 不过这样就会导致一个问题: customRef
作为 props
传递,进入了 WrappedComponent
组件内部,此时 customRef
还没有作为ref
属性挂载到任何 class组件
或者 DOM节点
上。
如果这时访问ref.current
会得到null
不过这也更方便的让我们转发ref
到组件内部的DOM
节点
// Bar.jsx
render() {
return (
<div ref={this.props.customRef}>
name: {this.props.name}
<br/>
age: {this.props.age}
<br/>
hobby: {this.props.hobby}
</div>
);
}
其实用最开始的写法让customRef
作为ref
属性挂载到组件上,在组件内部使用 React.forwardRef
一样能将ref
转发到组件内部的DOM
节点上。
最后
没有最后啦,Context 后面再补吧😁,没想到写这个挺费时间的,我要赶紧学后面的了,大家加油!
转载自:https://juejin.cn/post/6985068487479656461