React如何避免重渲染_javascript技巧

这篇文章主要介绍了react如何避免重渲染,现在分享给大家,也可以给有需要的朋友做个参考

组件的重新渲染

我们可以在 React 组件中的 props 和 state 存放任何类型的数据,通过改变 props 和 state,去控制整个组件的状态。当 props 和 state 发生变化时,React 会重新渲染整个组件,组件重新渲染的过程可简化如下图:

React如何避免重渲染_javascript技巧

译者之前对diff的理解是,对于一个改变 props 的组件,diff能自动计算出组件内部DOM树的不同,然后经过对比,找出真正变化的DOM节点,对变化部分进行渲染。这个是错误的理解,diff算法只是用来计算出改变状态或 props的组件/虚拟节点,而这个组件/虚拟节点,无论多大,它都会重新渲染。

假设有一个渲染完成的组件,如下图:

立即学习“Java免费学习笔记(深入)”;

React如何避免重渲染_javascript技巧

接下来因为状态改变,需要重新渲染下图的绿色的节点,如下图:

React如何避免重渲染_javascript技巧

一般的想法是只需要更新下面的三个绿色节点就能够完成组件的更新

React如何避免重渲染_javascript技巧

然而!只要组件的 props 或 state 发生了变化就会重新渲染整个组件,因此除了上述的三个绿色节点以外,还需要重新渲染所有的黄色的节点

React如何避免重渲染_javascript技巧

除了必要渲染的三个节点外,还渲染了其他不必要渲染的节点,这对性能是一个很大的浪费。如果对于复杂的页面,这将导致页面的整体体验效果非常差。因此要提高组件的性能,就应该想尽一切方法减少不必要的渲染。

shouldComponentUpdate

shouldComponentUpdate这个函数会在组件重新渲染之前调用,函数的返回值确定了组件是否需要重新渲染。函数默认的返回值是 true,意思就是只要组件的 props 或者 state 发生了变化,就会重新构建 virtual DOM,然后使用 diff 算法进行比较,再接着根据比较结果决定是否重新渲染整个组件。函数的返回值为 false 表示不需要重新渲染。

函数默认返回为 true.

PureRenderMixin

React 官方提供了 PureRenderMixin 插件,插件的功能就是在不必要的情况下让函数 shouldComponentUpdate 返回 false, 使用这个插件就能够减少不必要的重新渲染,得到一定程度上的性能提升,其使用方法如下:

  import PureRenderMixin from 'react-addons-pure-render-mixin';  class FooComponent extends React.Component {   constructor(props) {    super(props);    this.shouldComponentUpdate = PureRenderMixin.shouldComponentUpdate.bind(this);   }   render() {    return 

foo

;   }  }

登录后复制

我们需要在组件中重写 shouldComponentUpdate,PureRenderMixin源码中对PureRenderMixin.shouldComponentUpdate的定义是这样

  shouldComponentUpdate(nextProps, nextState) {    return shallowCompare(this, nextProps, nextState);  }

登录后复制

重写的方法里面根据组件的目前的状态和组件接下来的状态进行浅比较,如果组件的状态发生变化则返回结果为 false,状态没有发生变化则返回结果为 true

  shouldComponentUpdate(nextProps, nextState) {    return !shallowEqual(this.props, nextProps) ||        !shallowEqual(this.state, nextState);  }

登录后复制

在 React 的最新版本里面,提供了 React.PureComponent 的基础类,而不需要使用这个插件。

译者注:所以在一个较大的组件决定重渲染的时候,我们可以在每一个子组件中绑定新的shouldComponentUpdate方法,这样可以减少子组件重新渲染的次数。

我们自己可以重写 shouldComponentUpdate 这个函数,使得其能够对任何事物进行比较,也就是深比较(通过一层一层的递归进行比较),深比较是很耗时的,一般不推荐这么干,因为要保证比较所花的时间少于重新渲染的整个组件所花的时间,同时为了减少比较所花的时间我们应该保证 props 和 state 尽量简单,不要把不必要的属性放入 state,能够由其他属性计算出来的属性也不要放入 state 中。

Immutable.js

对于复杂的数据的比较是非常耗时的,而且可能无法比较,通过使用 Immutable.js 能够很好地解决这个问题,Immutable.js 的基本原则是对于不变的对象返回相同的引用,而对于变化的对象,返回新的引用。因此对于状态的比较只需要使用如下代码即可:

  shouldComponentUpdate() {    return ref1 !== ref2;  }

登录后复制

同样需要我们在子组件中将shouldComponentUpdate方法重写。

Pure Component

如果一个组件只和 props 和 state 有关系,给定相同的 props 和 state 就会渲染出相同的结果,那么这个组件就叫做纯组件,换一句话说纯组件只依赖于组件的 props 和 state,下面的代码表示的就是一个纯组件。

   render() {     return (       

           {this.state.rows}       

     );  }

登录后复制

如果某个子组件的 props 是固定的不会发生变化,我们叫做无状态组件。在这个组件里面使用 pureRenderMixin 插件,能够保证 shouldComponentUpdate 的返回一直为 false。所以,分清纯组件和无状态组件,在无状态组件中重写shouldComponentUpdate方法是最好的选择。

key

在写动态子组件的时候,如果没有给动态子项添加key prop,则会报一个警告。这个警告指的是,如果每一个子组件是一个数组或者迭代器的话,那么必须有一个唯一的key prop,那么这个key prop是做什么的呢?
我们想象一下,假如需要渲染一个有5000项的成绩排名榜单,而且每隔几秒就会更新一次排名,其中大部分排名只是位置变了,还有少部分是完全更新了,这时候key就发挥作用了,它是用来标识当前的唯一性的props。现在尝试来描述这一场景

  [{   sid: '10001',   name: 'sysuzhyupeng'  }, {   sid: '10008',   name: 'zhyupeng'  }, {   sid: '120000',   name: 'yupeng'  }]

登录后复制

其中sid是学号,那么我们来实现成绩排名的榜单

  import React from 'react';  function Rank({ list }){    return (     

登录后复制       {list.map((entry, index)=>(         {entry.name}       ))}         )  }

我们把key设成了序号,这么做的确不会报警告了,但这样是非常低效的做法,这个key是用来做virtual Dom diff的,上面的做法相当于用了一个随机键,那么不论有没有相同的项,更新都会重新渲染。

正确的做法非常简单,只需要把key的内容换成sid就可以了。

那么还有另一个问题,当key相同的时候,React会怎么渲染呢,答案是只渲染第一个相同key的项,且会报一个警告。

相关推荐:

代码详解React Js 微信分享封装

在React里使用Vuex的具体步骤

react怎样实现页面代码分割和按需加载

以上就是React如何避免重渲染_javascript技巧的详细内容,更多请关注【创想鸟】其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至253000106@qq.com举报,一经查实,本站将立刻删除。

发布者:PHP中文网,转转请注明出处:https://www.chuangxiangniao.com/p/2773943.html

(0)
上一篇 2025年3月8日 14:00:30
下一篇 2025年3月2日 01:43:09

AD推荐 黄金广告位招租... 更多推荐

相关推荐

  • Pastate.js 之响应式 react state 管理框架

    本篇文章给大家分享的内容是关于pastate.js 之响应式 react state 管理框架 ,有着一定的参考价值,有需要的朋友可以参考一下 Pastate 简介 Pastate 是什么 Pastate 是一个响应式 react stat…

    2025年3月8日 编程技术
    200
  • Pastate.js 响应式框架之多组件应用

    本篇文章给大家分享的内容是关于pastate.js 响应式框架之多组件应用  ,有着一定的参考价值,有需要的朋友可以参考一下 这是 pastate 系列教程的第二章,欢迎关注,持续更新。 这一章,我们在上一章的 state 结构中添加多一些…

    2025年3月8日 编程技术
    200
  • Pastate.js 响应式框架之数组渲染与操作

    本篇文章给大家分享的内容是关于pastate.js 响应式框架之数组渲染与操作  ,有着一定的参考价值,有需要的朋友可以参考一下 这是 Pastate.js 响应式 react state 管理框架系列教程的第三章,欢迎关注,持续更新。 这…

    2025年3月8日 编程技术
    200
  • Pastate.js 响应式 react 框架之表单渲染与操作

    本篇文章给大家分享的内容是关于pastate.js 响应式 react 框架之表单渲染与操作  ,有着一定的参考价值,有需要的朋友可以参考一下 这是 Pastate.js 响应式 react state 管理框架系列教程的第四章,欢迎关注,…

    2025年3月8日 编程技术
    200
  • Pastate.js 响应式 react 框架之 模块化

    本篇文章给大家分享的内容是关于pastate.js 响应式 react 框架之 模块化  ,有着一定的参考价值,有需要的朋友可以参考一下 这是 Pastate.js 响应式 react state 管理框架系列教程,欢迎关注,持续更新。 P…

    2025年3月8日 编程技术
    200
  • JavaScript创建对象

    本篇文章给大家分享的内容是关于javascript创建对象,有着一定的参考价值,有需要的朋友可以参考一下 调用系统的构造函数 创建一个最简单的方式就是创建一个Object的实例,如下:     // 1. 调用系统的构造函数    var …

    编程技术 2025年3月8日
    200
  • 详解js中的遍历 问题

    本篇文章给大家分享的内容是关于详解js中的遍历 问题,有着一定的参考价值,有需要的朋友可以参考一下 实例属性和原型属性 JavaScript中对象的属性分为两种: 数据属性 和 访问器属性 。 根据具体的上下文环境的不同,又可以将属性分为:…

    编程技术 2025年3月8日
    200
  • 使用node.js构建命令行工具

    本篇文章给大家分享的内容是使用node.js构建命令行工具 ,有着一定的参考价值,有需要的朋友可以参考一下 工具说明 inquirer.js:一个封装了常用命令行交互的node.js模块,通过该模块可以很方便地构建一个新的命令行应用。 sh…

    编程技术 2025年3月8日
    200
  • Vue.js提升必知的几点总结

    这次给大家带来Vue.js提升必知的几点总结,Vue.js提升的注意事项有哪些,下面就是实战案例,一起来看一下。 第一招:化繁为简的Watchers 场景还原: created(){ this.fetchPostList()},watch:…

    编程技术 2025年3月8日
    200
  • 在vuejs中使用v-show不起作用的原因有哪些

    这次给大家带来在vuejs中使用v-show不起作用的原因有哪些,在vuejs中使用v-show的注意事项有哪些,下面就是实战案例,一起来看一下。 1.官网概念描述 v-if 是’真正的’条件渲染,因为它会确保在切换…

    编程技术 2025年3月8日
    200

发表回复

登录后才能评论