React 组件通讯

TIP

组件是独立且封闭的单元,默认情况下,只能使用组件自己的数据。在组件化过程中,我们将一个完整的功能 拆分成多个组件,以更好的完成整个应用的功能。而在这个过程中,多个组件之间不可避免的要共享某些数据 。为了实现这些功能,就需要打破组件的独立封闭性,让其与外界沟通。这个过程就是组件通讯

props

  • 组件是封闭的,要接受外部数据通过 props 来实现
  • props 作用:接收传递给组件的数据
  • 传递数据:给组件标签添加属性
  • 接收数据:函数组件通过参数 props接收数据,类组件通过 this.props 接收数据

函数组件通讯

子组件

function Hello(props) {
  console.log(props)
  return <div>接收数据:{props.name}</div>
}

父组件

<Hello name="张三" age={20}></Hello>

类组件通讯

子组件

class Hello extends React.Component {
  render() {
    return <div>接收数据:{this.props.age}</div>
  }
}

父组件

<Hello name="张三" age={19} />

props 的特点

  • 可以给组件传递任意类型的数据
  • props 是只读的,不允许修改 props 的数据
  • 注意:在类组件中时候的时候,需要把props传递给 super()否则构造函数无法获取到 props
class Hello extends React.Component {
  constructor(props) {
    // 将props传递给父类构造函数
    super(props)
  }
  render() {
    return <div>接收数据{this.props.age}</div>
  }
}

组件通讯三种方式

  • 父传子
  • 子传父
  • 非父子

父传子

  1. 父组件提供要传递的 state 数据
  2. 给子组件标签添加属性,值为 state 中的数据
  3. 子组件中通过 props 接收父组件中传递的数据

父组件提供数据并且传递给子组件

class Parent extends React.Component {
  state = { lastName: '王' }
  render() {
    return (
      <div>
        传递数据给子组件:
        <Child name={this.state.lastName} />
      </div>
    )
  }
}

子组件接收数据

function Child(props) {
  return <div>子组件接收到数据:{props.name}</div>
}

子传父

思路:利用回调函数,父组件提供回调,子组件调用,将要传递的数据作为回调函数的参数。

  1. 父组件提供一个回调函数(用于接收数据)
  2. 将该函数作为属性的值,传递给子组件
  3. 子组件通过 props 调用回调函数
  4. 将子组件的数据作为参数传递给回调函数

父组件提供函数并且传递给字符串

class Parent extends React.Component {
  getChildMsg = (msg) => {
    console.log('接收到子组件数据', msg)
  }
  render() {
    return (
      <div>
        子组件:
        <Child getMsg={this.getChildMsg} />
      </div>
    )
  }
}

子组件接收函数并且调用

class Child extends React.Component {
    state = { childMsg: 'React' }
    handleClick = () => {
    	this.props.getMsg(this.state.childMsg)
    }
    return (
    	<button onClick={this.handleClick}>点我,给父组件传递数据</button>
    )
}

注意:回调函数中 this 指向问题!

非父子

  • 将共享状态提升到最近的公共父组件中,由公共父组件管理这个状态
  • 思想:状态提升
  • 公共父组件职责:
    • 提供共享状态
    • 提供操作共享状态的方法
  • 要通讯的子组件只需通过 props 接收状态或操作状态的方法

状态提升前

状态提升之后

组件通讯-context

基本概念

多层传递:App 组件传递数据给 Child 组件,该如何处理 处理方式:使用 props 一层层组件往下级传递(繁琐且数据容易丢失) 更好的方式使用:Conetxt 作用跨组件传递数据(比如:token、用户信息、系统主题、多语言切换)

实现思路

  • 调用 React.createContext() 创建 Provider(提供数据) 和 Consumer(消费数据) 两个组件
const { Provider, Consumer } = React.createContext()
  • 使用 Provider 组件作为父节点
<Provider>
  <div className="App">
    <Child1 />
  </div>
</Provider>
  • 设置 value 属性,表示要传递的数据
<Provider value="pink">
  • 调用 Consumer 组件接收数据
<Consumer>{(data) => <span>data参数表示接收到的数据 -- {data}</span>}</Consumer>

总结

  1. 如果两个组件是远方亲戚(比如,嵌套多层)可以使用 Context 实现组件通讯
  2. Context 提供了两个组件:Provider 和 Consumer
  3. Provider 组件:用来提供数据
  4. Consumer 组件:用来消费数据

props 深入

children 属性

  • children 属性:表示该组件的子节点,只要组件有子节点,props 就有该属性
  • children 属性与普通的 props 一样,值可以是任意值(文本、React 元素、组件、函数等)
function Hello(props) {
  return <div>该组件的子节点:{props.children}</div>
}

;<Hello>我是子节点</Hello>

props 校验

目的:校验接收的 props 的数据类型,增加组件的健壮性。 对于组件来说:props 是外来的,无法的保证组件使用者传入什么数据格式。 如果传入的数据格式不对,可能会导致组件内部内部错误组件的使用者不能很明确的知道错误原因

// 成员A创建组件App
function App(props) {
  const arr = props.colors
  const list = arr.map((item, index) => <li key={index}>{item}</li>)
  return <ul>{list}</ul>
}
// 成员B使用App组件
<App colors={19}>

props 校验允许在创建组件的时候,就约定 props 的格式、类型等

App.propTypes = {
  colors: PropTypes.array
}

作用:规定接收的 props 的类型必须为数组,如果不是数组就会报错,增加组件的健壮性。

使用步骤

  1. 安装包 prop-typesyarn add prop-types / npm i props-types
  2. 导入 prop-types
  3. 使用组件名.propTypes = {} 来给组件的props添加校验规则
  4. 校验规则通过 PropTypes 对象来指定
import PropTypes from 'prop-types'
function App(props) {
  return <h1>Hi, {props.colors}</h1>
}
App.propTypes = {
  // 约定colors属性为array类型
  // 如果类型不对,则报出明确错误,便于分析错误原因
  colors: PropTypes.array
}

约束规则

  1. 常见类型:arrayboolfuncnumberobjectstring
  2. React 元素类型:element
  3. 必填项:isRequired
  4. 特定结构的对象:shape({ })
// 常见类型
optionalFunc: PropTypes.func,
// 必选
requiredFunc: PropTypes.func.isRequired,
// 特定结构的对象
optionalObjectWithShape: PropTypes.shape({
	color: PropTypes.string,
	fontSize: PropTypes.number
})

props 默认值

场景:分页组件,每页显示条数 作用:给 props 设置默认值,在未传入 props 时生效

function App(props) {
    return (
        <div>
            此处展示props的默认值:{props.pageSize}
        </div>
    )
}
// 设置默认值
App.defaultProps = {
	pageSize: 10
}
// 不传入pageSize属性
<App />

类的静态属性 static

  • 实例成员: 通过实例调用的属性或者方法,,叫做实例成员(属性或者方法)
  • 静态成员:通过类或者构造函数本身才能访问的属性或者方法
class Person {
   name = 'zs',
   static age = 18
   sayHi() {
       console.log('哈哈')
   }
   static goodBye() {
       console.log('byebye')
   }
}

const p = new Person()

console.log(p.name)
p.sayHi()

console.log(Person.age)
Person.goodBye()