coleava / me

1 stars 0 forks source link

React Component和pureComponent #13

Open coleava opened 3 years ago

coleava commented 3 years ago

PureComponent通过prop和state的浅比较来实现shouldComponentUpdate,某些情况下可以用PureComponent提升性能

浅比较(shallowEqual),即react源码中的一个函数,然后根据下面的方法进行是不是PureComponent的判断

if (this._compositeType === CompositeTypes.PureClass) {
  shouldUpdate = !shallowEqual(prevProps, nextProps) || ! shallowEqual(inst.state, nextState);
}

哪些情况下可以使用PureComponent来提升性能 例1: IndexPage组件,设置了一个state是isShow,通过一个按钮点击可以改变它的值,结果是:初始化的时候输出的是constructor,render,而第一次点击按钮,会输出一次render,即重新渲染了一次,界面也会从显示false变成显示true, --- 当这个组件是继承自PureComponent的时候,再点击的时,不会再输出render,即不会再重新渲染了 --- 而当这个组件是继承自Component时,还是会输出render,还是会重新渲染,这时候就是PureComponent内部做了优化的体现

同理也适用于string,number等基本数据类型,因为基本数据类型,值改变了就算改变了

import React, { PureComponent } from 'react';

class IndexPage extends PureComponent{
  constructor() {
    super();
    this.state = {
      isShow: false
    };
    console.log('constructor');
  }
  changeState = () => {
    this.setState({
      isShow: true
    })
  };
  render() {
    console.log('render');
    return (
      <div>
        <button onClick={this.changeState}>点击</button>
        <div>{this.state.isShow.toString()}</div>
      </div>
    );
  }
}

例2 :

class IndexPage extends PureComponent {
  constructor() {
    super();
    this.state = {
      arr: ['1'],
      obj: {
        name: "zlq"
      }
    };
    console.log('constructor');
  }
  //PureComponent中页面会警告,不建议在PureComponent中使用
  // shouldComponentUpdate(){
  //   return false;
  // }
  changeState = () => {
    let { arr } = this.state;
    arr.push('2') 
    var obj = {
      name: "zl"
    }
    this.setState({
      // obj
      // obj:{name:"Z"}
      arr // pureComponent下不会重新render
      // arr: [...arr, '2']   **这里使用了扩展运算符 改变了arr的引用**
    })
  };
  render() {
    console.log('render');
    let { arr, obj } = this.state
    return (
      <div>
        <button onClick={this.changeState}>点击</button>
        <div>
          {arr.map((item) => {
            return item;
          })}
          {obj.name}
        </div>
      </div>
    );
  }
}

export default IndexPage;

PureComponent不仅会影响本身,而且会影响子组件,所以PureComponent最佳情况是展示组件

//父组件
import React, { PureComponent, Component } from 'react';
import Example from "../components/Example";

class IndexPage extends PureComponent{
  constructor() {
    super();
    this.state = {
      person: {
        name: 'sxt'
      }
    };
    console.log('constructor');
  }
  changeState = () => {
    let { person } = this.state;
    person.name = 'sxt2';
    this.setState({
      person
    })
  };
  render() {
    console.log('IndexPage render');
    const { person } = this.state;
    return (
      <div>
        <button onClick={this.changeState}>点击</button>
        <Example person={person} />
      </div>
    );
  }
}
//子组件
import React, { Component } from 'react';
class Example extends Component {
  render() {
    console.log('example render');
    const { person } = this.props;
    return(
      <div>
        {person.name}
      </div>
    );
  }
}
  1. 父组件继承PureComponent,子组件继承Component时:下面的结果初始化时输出为constructor,IndexPage render,example render,但是当我们点击按钮时,界面没有变化,因为这个this.state.person对象的引用没有改变,只是改变了它里面的属性值所以尽管子组件是继承Component的也没有办法渲染,因为父组件是PureComponent,父组件根本没有渲染,所以子组件也不会渲染
  2. 父组件继承PureComponent,子组件继承PureComponent时:因为渲染在父组件的时候就没有进行,相当于被拦截了,所以子组件是PureComponent还是Component根本不会影响结果,界面依旧没有变化
  3. 父组件继承Component,子组件继承PureComponent时:结果和我们预期的一样,即初始化是会输出constructor,IndexPage render,example render,但是点击的时候只会出现IndexPage render,因为父组件是Component,所以父组件会渲染,但是当父组件把值传给子组件的时候,因为子组件是PureComponent,所以它会对prop进行浅比较,发现这个person对象的引用没有发生变化,所以不会重新渲染,而界面显示是由子组件显示的,所以界面也不会变化
  4. .父组件继承Component,子组件继承Component时:初始化是会输出constructor,IndexPage render,example render,当我们第一次点击按钮以后,界面发生变化,后面就不再改变,因为我们一直把它设置为sxt2,但是每点击一次都会输出IndexPage render,example render,因为每次不管父组件还是子组件都会渲染
  5. 所以正如下面第三条说的,如果state和prop一直变化的话,还是建议使用Component,并且PureComponent的最好作为展示组件

若是数组和对象等引用类型,则要引用不同,才会渲染

继承自Component的组件,若是shouldComponentUpdate返回false,就不会渲染了,继承自PureComponent的组件不用我们手动去判断prop和state,所以在PureComponent中使用shouldComponentUpdate会有如下警告: IndexPage has a method called shouldComponentUpdate(). shouldComponentUpdate should not be used when extending React.PureComponent. Please extend React.Component if shouldComponentUpdate is used.

PureComponent使用注意事项:即如果改变的是数组或对象内部的某个状态,则必须使用解构赋值语法重新setState,否则不会更新 https://img2020.cnblogs.com/blog/1291869/202004/1291869-20200404103619466-1397783531.png