调用方法:1、类组件中的调用可以利用React.createRef()、ref的函数式声明或props自定义onRef属性来实现;2、函数组件、Hook组件中的调用可以利用useImperativeHandle或forwardRef抛出子组件ref来实现。
 
  
 
 
 
 【程序员必备开发工具推荐】Apifox一款免费API管理工具Apifox = Postman + Swagger + Mock + JMeter 
  
 
本教程操作环境:Windows7系统、react18版、Dell G3电脑。
 
在React中,我们经常在子组件中调用父组件的方法,一般用props回调即可。但是有时候也需要在父组件中调用子组件的方法,通过这种方法实现高内聚。有多种方法,请按需服用。
 
类组件中 1、React.createRef()  1
 2
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
   import React, { Component } from 'react';
 
 class Sub extends Component {
   callback() {
     console.log('执行回调');
   }
   render() {
     return <div>子组件</div>;
   }
 }
 
 class Super extends Component {
   constructor(props) {
     super(props);
     this.sub = React.createRef();
   }
   handleOnClick() {
     this.sub.callback();
   }
   render() {
     return (
       <div>
         <Sub ref={this.sub}></Sub>
       </div>
     );
   }
 }
  
 
2、ref的函数式声明 优点:ref写法简洁 缺点:使用了HOC的子组件不可用,无法指向真是子组件(同上) 使用方法和上述的一样,就是定义ref的方式不同。
 
 1
 2
 3
 4
 5
   ...
 
 <Sub ref={ref => this.sub = ref}></Sub>
 
 ...
  
 
3、使用props自定义onRef属性 优点:假如子组件是嵌套了HOC,也可以指向真实子组件。 缺点:需要自定义props属性  1
 2
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
   import React, { Component } from 'react';
 import { observer } from 'mobx-react'
 
 @observer
 class Sub extends Component {
     componentDidMount(){
     // 将子组件指向父组件的变量
         this.props.onRef && this.props.onRef(this);
     }
     callback(){
         console.log("执行我")
     }
     render(){
         return (<div>子组件</div>);
     }
 }
 
 class Super extends Component {
     handleOnClick(){
        // 可以调用子组件方法
         this.Sub.callback();
     }
     render(){
         return (
           <div>
             <div onClick={this.handleOnClick}>click</div>
             <Sub onRef={ node => this.Sub = node }></Sub>    
           </div>)
     }
 }
  
 
函数组件、Hook组件 1、useImperativeHandle 优点: 1、写法简单易懂 2、假如子组件嵌套了HOC,也可以指向真实子组件 缺点: 1、需要自定义props属性 2、需要自定义暴露的方法  1
 2
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
   import React, { useImperativeHandle } from 'react';
 import { observer } from 'mobx-react'
 
 
 const Parent = () => {
   let ChildRef = React.createRef();
 
   function handleOnClick() {
     ChildRef.current.func();
   }
 
   return (
     <div>
       <button onClick={handleOnClick}>click</button>
       <Child onRef={ChildRef} />
     </div>
   );
 };
 
 const Child = observer(props => {
   //用useImperativeHandle暴露一些外部ref能访问的属性
   useImperativeHandle(props.onRef, () => {
     // 需要将暴露的接口返回出去
     return {
       func: func,
     };
   });
   function func() {
     console.log('执行我');
   }
   return <div>子组件</div>;
 });
 
 export default Parent;
  
 
2、forwardRef 使用forwardRef抛出子组件的ref
 
这个方法其实更适合自定义HOC。但问题是,withRouter、connect、Form.create等方法并不能抛出ref,假如Child本身就需要嵌套这些方法,那基本就不能混着用了。forwardRef本身也是用来抛出子元素,如input等原生元素的ref的,并不适合做组件ref抛出,因为组件的使用场景太复杂了。
 
 1
 2
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
   import React, { useRef, useImperativeHandle } from 'react';
 import ReactDOM from 'react-dom';
 import { observer } from 'mobx-react'
 
 const FancyInput = React.forwardRef((props, ref) => {
   const inputRef = useRef();
   useImperativeHandle(ref, () => ({
     focus: () => {
       inputRef.current.focus();
     }
   }));
 
   return <input ref={inputRef} type="text" />
 });
 
 const Sub = observer(FancyInput)
 
 const App = props => {
   const fancyInputRef = useRef();
 
   return (
     <div>
       <FancyInput ref={fancyInputRef} />
       <button
         onClick={() => fancyInputRef.current.focus()}
       >父组件调用子组件的 focus</button>
     </div>
   )
 }
 
 export default App;
  
 
总结  
父组件调子组件函数有两种情况
 
子组件无HOC嵌套:推荐使用ref直接调用 有HOC嵌套:推荐使用自定义props的方式