当前位置: 首页 > article >正文

React父组件怎么调用子组件的方法

调用方法:1、类组件中的调用可以利用React.createRef()、ref的函数式声明或props自定义onRef属性来实现;2、函数组件、Hook组件中的调用可以利用useImperativeHandle或forwardRef抛出子组件ref来实现。

React父组件怎么调用子组件的方法

【程序员必备开发工具推荐】Apifox一款免费API管理工具Apifox = Postman + Swagger + Mock + JMeter

本教程操作环境:Windows7系统、react18版、Dell G3电脑。

在React中,我们经常在子组件中调用父组件的方法,一般用props回调即可。但是有时候也需要在父组件中调用子组件的方法,通过这种方法实现高内聚。有多种方法,请按需服用。

类组件中


1、React.createRef()
  • 优点:通俗易懂,用ref指向。

  • 缺点:使用了HOC的子组件不可用,无法指向真是子组件

    比如一些常用的写法,mobx的@observer包裹的子组件就不适用此方法。

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的方式

http://www.kler.cn/a/132581.html

相关文章:

  • oracle19c开机自启动
  • 【ASR技术】WhisperX安装使用
  • Linux最深刻理解页表于物理内存
  • Python数据分析NumPy和pandas(三十五、时间序列数据基础)
  • Failed to create a temp file - Jenkins 无法创建任务
  • Excel单元格中自适应填充多图
  • c语言免杀火绒
  • centos7 网卡聚合bond0模式配置
  • vs2017打开工程提示若要解决此问题,请使用以下选择启动 Visual Studio 安装程序: 用于 x86 和 x64 的 Visual C++ MFC
  • Buildroot 添加 Qt 支持
  • 2023.11.18 - hadoop之zookeeper分布式协调服务
  • JVM——运行时数据区(程序计数器+栈)
  • django理解02 前后端分离中的问题
  • 栈和队列的初始化,插入,删除,销毁。
  • 详谈动态规划问题并解最大子数组和
  • 构建自定义ChatGPT,微软推出Copilot Studio
  • 使用 `spring-statemachine-redis` 实现状态机
  • Xrdp+Cpolar实现远程访问Linux Kali桌面
  • SecureCRT 9.4.2 for Mac
  • 【数据结构】图的存储结构(邻接矩阵)
  • 云计算的发展趋势
  • JVM虚拟机:CMS垃圾回收器的日志分析
  • Hive函数
  • 基于操作系统讨论Java线程与进程、浅谈Go的线程与管程
  • excel怎么能锁住行 和/或 列的自增长,保证粘贴公式的时候不自增长或者只有部分自增长
  • 查询/新增/删除 在B+树上的动态体现