likes
comments
collection
share

React 中的 forwardRef 究竟该怎么用?

作者站长头像
站长
· 阅读数 67

React 中的 forwardRef

React 中的 forwardRef 究竟该怎么用?

前言

最近在学react, 一直对forwardRefContext这两个认识不太清楚,现在终于搞懂了,于是想写一篇文章记录一下。

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转发的流程:

  1. 创建了一个ref
  2. 将其挂载到 组件上这个组件是通过React.forwardRef创建出来的, 注意这里很关键,后面细说
  3. 组件Foo接收到了一个ref,于是将它转发到DOM节点input
  4. ref如愿的挂载到内部节点input
  5. 现在this.input.current保存着对节点input的引用
  6. 点击按钮, 现在可以很轻松的获取Foo内部节点的value以及获取其焦点

React 中的 forwardRef 究竟该怎么用?

细节补充

之前说过,步骤2很关键,这是因为 ref 的值根据节点的类型而有所不同:

  1. ref 属性用于 HTML 元素时,接收底层 DOM 元素作为其 current 属性。

  2. ref 属性用于自定义 class 组件时,ref 接收组件实例作为其 current 属性。

  3. 不能在函数组件上使用 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 组件挂载后生命周期函数打印的

第二条打印是点击按钮后打印的,证明确实只是挂载到了组件实例上。后面的高阶组件中还会出现类似的问题。

React 中的 forwardRef 究竟该怎么用?

对于第三个,不能在函数组件上使用 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 究竟该怎么用?

一般函数式组件都是用React.forwardRef包装一下然后返回出去的, 函数式组件本来就是一个render函数,不过在被React.forwardRef包装后就多了一个ref属性了。

// 将我们的函数式组件改造成这个样子.
const Baz = React.forwardRef((props, ref) => {
  return (
    <div>
      啊?
    </div>
  );
})

此时的ref还没有被挂载,所以访问ref.current会得到null,不过总算不会报错了

React 中的 forwardRef 究竟该怎么用?

函数式组件只是将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;

React 中的 forwardRef 究竟该怎么用?

在高阶组件中转发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再次登场,我们可以使用 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} />
  ))
  
}
  1. 我们将 LogProps组件作为render函数的返回值,这样渲染结果还是不变,然后将传入的ref转发到 LogProps组件的自定义属性 customRef上。注意这里一定要转发到自定义属性,如果转发到 ref属性上最终还是会被挂载到 LogProps上,等于转了一圈又回到了原地...
  2. 所有的属性都被传递到props
  3. props中的 customRef提取出来,最终传递到WrappedComponentref属性上。

最后ref被成功转发到被包裹的组件WrappedComponent上。

点击按钮,现在来控制台看看

React 中的 forwardRef 究竟该怎么用? ref 被成功的挂载到目标组件上。

细心的读者可能已经发现logProps.js转发ref还有另一种写法, 就是 3.5 那样,不用抽离 customRef,仍然将 customRef 作为props 透传下去, 不过这样就会导致一个问题: customRef作为 props传递,进入了 WrappedComponent组件内部,此时 customRef还没有作为ref属性挂载到任何 class组件或者 DOM节点上。

如果这时访问ref.current会得到null

React 中的 forwardRef 究竟该怎么用?

不过这也更方便的让我们转发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>
  );
}

React 中的 forwardRef 究竟该怎么用?

其实用最开始的写法让customRef作为ref属性挂载到组件上,在组件内部使用 React.forwardRef一样能将ref转发到组件内部的DOM节点上。

最后

没有最后啦,Context 后面再补吧😁,没想到写这个挺费时间的,我要赶紧学后面的了,大家加油!

转载自:https://juejin.cn/post/6985068487479656461
评论
请登录