React 的函数组件和 Class 组件

React 的函数组件和 Class 组件_第1张图片

组件是 React 里非常重要的组成部分,其分为函数组件和 Class 组件。本文就简单说明这两种组件定义方式的由来。

例子

让我们先从一个简单的需求开始。定义一个加减器,就是用来做简单的加减法。使用 JSX 语法我们可以写成这样:

let number = 0
let add = () => {
    number += 1
    render()
}

let minus = () => {
    number -= 1
    render()
}

let render = () => {
    ReactDOM.render(
        
// Adder1 {number}
, document.querySelector('#root')) } render()

函数组件

现在我们想要更多的加减器,那么可能会在 render() 函数里写很多个

...
,这样明显不好。

还记得 JSX 语法里的并不是真正的 HTML,而是虚拟 DOM,即 JS 代码,不信可看转译后的结果:

React 的函数组件和 Class 组件_第2张图片
转译结果

既然是 JS 代码,那么我们就可以用 JS 的方法来将其分块了。我们定义多个函数,来返回虚拟 DOM 不就可以完成分块了么?所以定义两个函数:

function App() {
  return (
    
// React.createElement(Adder1) // React.createElement(Adder2)
) } function Adder1() { return (
{number}
) } ...

虽然这里有个问题,变量 number 被 Adder1 和 Adder2 共享了。但是 React 的一个简单组件就诞生了,其本质就是一个函数。

props

React 的开发者很聪明,即然这个组件返回的是虚拟 DOM,那么正常的 DOM 应该要有属性才行,而函数的参数好像和这属性有着某种相关性。所以函数组件的一个特性被开发出来了:函数组件传入的参数(对象) 代表了该虚拟 DOM 的属性。例子:

function Adder (props) {
  return (
    

My name is {props.name}, age: {props.age}

) } function App() { return (
) }

state

那么自身的属性呢?很简单,在函数里面定义就好了:

function Adder1(props) {
  let name = 'hello'
  return (
    

My name is {name}, age: {props.age}

) }

Class 组件

好了,现在说说怎么去解决共享 number 的问题。像上面说的用自身属性试试:

function Adder1(obj) {
  let number = 0
  function add() {...}
  function minus() {...}
  return (
    
{number}
) } ...

虽然好像看起来没问题,但是每次修改值后我们都要重新 render 一下组件的,render 的时候需要执行 Adder1 里面的代码,number 被重置了。

React 的 Class 组件就了为了这样的问题而诞生的:

class Adder1 extends React.Component {
  constructor(props) {
    super(props)
    this.state = {
      number: 0
    }
  }
  
  add() {
    this.setState({
      number: this.state.number + 1
    })
  }
  minus() {
    this.setState({
      number: this.state.number - 1
    })
  }
  
  render() {
    return (
      
{this.state.number} {this.props.name}
) } }

注意

  1. 必须要继承 React.Component
  2. constructor 里要传入 props,并调用 super(props) ,因为这是 ES6 语法的规定
  3. this.state 用来存放自身属性,但是修改的时候要用 this.setState(newState),而不能直接this.state.number += 1
  4. 绑定方法的时候要绑定this,因为 React 会这样调用 this.add.call(undefined, ...)

setState

为了对更新进行优化,如果多次修改 this.state 会将大批量更新合并成一次更新。其实 this.state 的更新方法属于异步更新,只有全部改变完了再去更新。

像下面的代码只会更新一次:

this.setState({
    number: this.state.number - 1
})
this.setState({
    number: this.state.number - 1
}) // 只是一次 - 1,因为每二次的时候 this.state.number 还可能是 0

当然可以用回调的形式进行多次更新:

this.setState((state) => {
    return { number: state.number - 1 }
})
this.setState((state) => {
    return { number: state.number - 1 }
})

你可能感兴趣的:(React 的函数组件和 Class 组件)