所有分类
  • 所有分类
  • 后端开发

React每日一学:在React中,父组件如何调用子组件

500.jpg

React每日学习:在React中,如何调用父组件的子组件。这种方法在平时的项目中经常使用。今天我们来看看!

使用子组件是类组件 Refs;使用 React.createRef() 创建 Refs, Refs 可以通过 ref 当属性附加到元素节点时,我们可以通过 ref.current 访问附加元素节点。

剖析

文章中涉及 ref 该应用只是父组件调用子组件场景中的一种应用,不包括 ref 所有应用方法!

Class组件

自定义事件

Parent.js

import React, { Component } from 'react';

import Child from './Child';

class Parent extends Component {
  componentDidMount () {
    console.log(this.childRef)
  }
  handleChildEvent = (ref) => {
  	// 存储子组件的实例 this.childRef 中, 这样,就可以得到整个父组件
    this.childRef = ref
  }
  //按钮事件处理
  handleClick = () => {
  	// 通过子组件的实例调用组件中的方法
    this.childRef.sendMessage()
  }
  render () {
    return (
      <>
        <Child
          onChildEvent={this.handleChildEvent}
        />
        <button onClick={this.handleClick}>Trigger Child Event</button>
      </>
    );
  }
}

export default Parent;

Child.js

import React, { Component } from 'react';

class Child extends Component {
	///子组件完成挂载时, 将子组件的方法 this 作为参数传输到父组件的函数
  componentDidMount () {
  	// 在子组件中调用父组件的方法,并将当前实例传输到当前实例中
    this.props.onChildEvent(this)
  }
  // 子组件方法, 在父组件中触发
  sendMessage = () => {
    console.log('sending message')
  }
  render () {
    return ( <div>Child</div> );
  }
}

export default Child;

2. 使用 React.createRef()

ParentCmp.js

import React, { Component } from 'react';

import ChildCmp from './ChildCmp';

export default class ParentCmp extends Component {
  constructor(props) {
    super(props)
    // 创建Ref
    this.childRef = React.createRef()
  }
  // 按钮事件
  handleClick = () => {
  	// 直接通过 this.childRef.current 拿到子组件的例子
    this.childRef.current.sendMessage()
  }
  render () {
    return (
      <>
        <ChildCmp ref={this.childRef} />
        <button onClick={this.handleClick}>Trigger Child Event</button>
      </>
    );
  }
}

子组件是一个普通的组件

ChildCmp.js

import React, { Component } from 'react';

export default class ChildCmp extends Component {
  sendMessage = () => {
    console.log('sending message')
  }
  render () {
    return 'Child';
  }
}

3. 使用回调Refs

回调 Refs 是另一种设置 Ref 它可以帮助你更精细地控制何时 refs 被设置和解除。

不同于传递 createRef() 创建的 ref 属性,需要传递函数。

访问 Ref 不需要的时候 current。

ParentCmp.js

import React, { Component } from 'react';

import ChildCmp from './ChildCmp';

export default class ParentCmp extends Component {
  constructor(props) {
    super(props)
    // 创建 Ref,不通过 React.createRef()
    this.childRef = null
  }
  // 设置 Ref
  setChildRef = (ref) => {
    this.childRef = ref
  }
  // 按钮事件
  handleClick = () => {
    // 直接通过 this.childRef 拿到子组件的例子
    this.childRef.sendMessage(`Trigger Child Event from Parent`)
  }

  render () {
    return (
      <>
        <ChildCmp ref={this.setChildRef} />
        <button onClick={this.handleClick}>Trigger Child Event</button>
      </>
    );
  }
}

子组件仍然是一个普通的组件

ChildCmp.js

import { Component } from 'react';

export default class ChildCmp extends Component {
  sendMessage = (message) => {
    console.log('sending message:', message)
  }
  render () {
    return 'Child';
  }
}

[注]比较自定义事件的方式,回调 Refs 更像是精简的自定义事件:

  • 自定义事件的名称变成了 ref

  • 子组件不需要手动绑定

Function组件

默认情况下,函数组件不能使用 ref 属性,因为它们没有例子。因此,以上两种方法是不可行的。

解决方案是使用 forwardRef 和 useImperativeHandle。

但是,它可以用于函数的内部 useRef 在组件中获取钩子 DOM 元素。

Parent.js

import React, { useRef } from 'react';

import Child from './Child';

const Parent = () => {
  // 通过 Hooks 创建 Ref
  const childRef = useRef(null)
  const handleClick = () => {
    childRef.current.sendMessage()
  }
  return (
    <>
      <Child
        ref={childRef}
      />
      <button onClick={handleClick}>Trigger Child Event</button>
    </>
  );
}
export default Parent;

Child.js

import React, { forwardRef, useImperativeHandle } from 'react';

const Child = forwardRef((props, ref) => {
  ///将子组件的方法 暴露给父组件
  useImperativeHandle(ref, () => ({
    sendMessage
  }))
  const sendMessage = () => {
    console.log('sending message')
  }
  return ( <div>Child</div> );
})
export default Child;

注:

上面的例子只是简单地展示了父子组件之间的调用方法。当然,在实际情况下,中子组件也可以有自己的方法 ref 指向自己内心的 DOM 元素,但这些原理是一样的。

补充:子组件调用父组件的方法

在子组件中调用父组件的setid方法。

父组件

 <NavBarX
   item={item}
   current={current}
   getBatchDetails={(id) => this.getBatchDetails(0, id)}
   setId={(id, callback) => this.setState({ id }, callback)}
   onRef={this.onNavBarXRef}
 />

子组件

this.props.setId(prePageId, () => {
   getBatchDetails(prePageId)
 })

总结

不同的元素类型会导致不同的元素类型 ref 的值不同:
① 当 ref 附加元素是 HTML 元素时,ref.current 指的是底部的DOM元素。
② 当 ref 附加元素是 class 组件时,ref.current 指向 class 组件实例。

注:在默认情况下,函数组不能使用 ref 属性, 因为他们没有例子。但结合 forwardRef 我们可以在函数组件上使用方法 ref,以后会介绍。

原文链接:https://www.icz.com/technicalinformation/web/seo/2023/05/9518.html,转载请注明出处~~~
0
注意:请收藏好网址www.icz.com,防止失联!站内免费资源持续上传中…!赞助我们
显示验证码
没有账号?注册  忘记密码?