React 父组件调用子组件的方法
在 React 中,我们经常在子组件中调用父组件的方法,一般用 props 回调即可。但是有时候也需要在父组件中调用子组件的方法,通过这种方法实现高内聚。有多种方法,请按需服用。
类组件中
React.createRef()
- 优点:通俗易懂,用 ref 指向。
- 缺点:使用了 HOC 的子组件不可用,无法指向真是子组件
比如一些常用的写法,mobx 的 @observer 包裹的子组件就不适用此方法。
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> ); } }
ref 的函数式声明
- 优点:ref 写法简洁
- 缺点:使用了 HOC 的子组件不可用,无法指向真是子组件(同上)
使用方法和上述的一样,就是定义 ref 的方式不同。
... <Sub ref={ref => this.sub = ref}></Sub> ...
使用 props 自定义 onRef 属性
- 优点:假如子组件是嵌套了 HOC,也可以指向真实子组件。
- 缺点:需要自定义 props 属性
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 组件
useImperativeHandle
- 优点: 1、写法简单易懂 2、假如子组件嵌套了 HOC,也可以指向真实子组件
- 缺点: 1、需要自定义 props 属性 2、需要自定义暴露的方法
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;
forwardRef
使用 forwardRef 抛出子组件的 ref
这个方法其实更适合自定义 HOC。但问题是,withRouter、connect、Form.create 等方法并不能抛出 ref,假如 Child 本身就需要嵌套这些方法,那基本就不能混着用了。forwardRef 本身也是用来抛出子元素,如 input 等原生元素的 ref 的,并不适合做组件 ref 抛出,因为组件的使用场景太复杂了。
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 的方式
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论