react组件渲染优化-类组件渲染优化

某些值没有达到改变的条件 或者没有必要渲染 ,但用户点击了;整个组件仍然是重新渲染了的。显然,这一次渲染是没有必要的。

例子:

javascript 复制代码
import React from 'react'
export default class App extends React.Component {
    constructor() {
    super();
    this.state = {
      counter: 1
    }
  }
  render() {
    console.log("App 渲染了");
    return (
      <div>
        <h1>App 组件</h1>
        <div>{this.state.counter}</div>
        <button onClick={() => this.setState({
          counter : 1
        })}>+1</button>
      </div>
    )
  }
}

方式一

使用shouldComponentUpdate来决定是否渲染

javascript 复制代码
import React, { Component } from 'react'
import { objectEqual } from "../utils/tools"

export default class ClassOptimize extends Component {

  constructor() {
    super();
    this.state = {
      counter: 1
    }
  }

  /**
   * 
   * @param {*} nextProps 新的 props
   * @param {*} nextState 新的 state
   * @returns 
   */
  shouldComponentUpdate(nextProps, nextState) {
    // shouldComponentUpdate会根据返回值来决定是否重新渲染
    // 默认是 true,要重新渲染
    // 如果返回 false,则不会重新渲染
    // 我们就需要将当前的 props 和 state 与新的 props 和 state 进行一个比较
    if (objectEqual(this.props, nextProps) && objectEqual(this.state, nextState)) {
      // 如果新旧 props 和 state 都是相同的,那么就返回 false,不需要重新渲染
      return false;
    }
    return true;
  }

  render() {
    console.log("ClassOptimize渲染了");
    return (
      <div>
        <h1>ClassOptimize 组件</h1>
        <div>{this.state.counter}</div>
        <button onClick={() => this.setState({
          counter: Math.floor(Math.random() * 3 + 1)
        })}>+1</button>
      </div>
    )
  }
}

/utils/tools.js

javascript 复制代码
/**
 * 对两个对象进行一个浅比较,看是否相等
 * obj1
 * obj2
 * 返回布尔值 true 代表两个对象相等, false 代表不想等
 */
export function objectEqual(obj1, obj2){
  for(let prop in obj1){
    if(!Object.is(obj1[prop],obj2[prop])){
      // 进入此 if,说明有属性值不相等
      // 只要有一个不相等,那么就应该判断两个对象不等
      return false;
    }
  }
  return true;
}
方式二

使用PureComponent 跳过不必要的重新渲染,如果值相同就不会渲染

javascript 复制代码
import React, { PureComponent } from 'react'

export default class ClassOptimize extends PureComponent {

  constructor() {
    super();
    this.state = {
      counter: 1
    }
  }

  render() {
    console.log("ClassOptimize渲染了");
    return (
      <div>
        <h1>ClassOptimize 组件</h1>
        <div>{this.state.counter}</div>
        <button onClick={() => this.setState({
          counter: 1
        })}>+1</button>
      </div>
    )
  }
}
方式三:扩展
javascript 复制代码
import React, { PureComponent } from 'react'
export default class ClassOptimize extends PureComponent {

  constructor() {
    super();
    this.state = {
      stu: ['张三', '李四']
    }
  }
  handleClick = () => {
    // 新增用户
    // 这样新增是没用的,视图不会更新,PureComponent进行的是浅层比较,它比较的是地址,看到地址一样就直接返回false阻止视图更新
    /* this.state.stu.push('王五')
    this.setState({
      stu: this.state.stu
    }) */
    // 这样新增
    const arr = [...this.state.stu]
    arr.push('王五')
    this.setState({
      stu: arr
    })
  }
  render() {
    const li = this.state.stu.map((item, index) => (<li key={index}>姓名:{item}</li>))
    return (
      <div>
        <h1>ClassOptimize 组件</h1>
        <button onClick={this.handleClick}>新增用户</button>
        <ul>
          {li}
        </ul>
      </div>
    )
  }
}

下一篇讲函数组件渲染优化

相关推荐
txinyu的博客几秒前
HTTP服务实现用户级窗口限流
开发语言·c++·分布式·网络协议·http
代码村新手1 分钟前
C++-类和对象(上)
开发语言·c++
rgeshfgreh9 分钟前
Spring事务传播机制深度解析
java·前端·数据库
全栈小精灵10 分钟前
Winform入门
开发语言·机器学习·c#
心静财富之门10 分钟前
退出 for 循环,break和continue 语句
开发语言·python
txinyu的博客12 分钟前
map和unordered_map的性能对比
开发语言·数据结构·c++·算法·哈希算法·散列表
Mr -老鬼28 分钟前
Rust适合干什么?为什么需要Rust?
开发语言·后端·rust
予枫的编程笔记31 分钟前
【Java集合】深入浅出 Java HashMap:从链表到红黑树的“进化”之路
java·开发语言·数据结构·人工智能·链表·哈希算法
ohoy37 分钟前
RedisTemplate 使用之Set
java·开发语言·redis
mjhcsp37 分钟前
C++ 后缀数组(SA):原理、实现与应用全解析
java·开发语言·c++·后缀数组sa