Your Guide to Efficient Re-rendering with React

简介: First, let’s understand the process of re-rendering in React. A React component translates raw data (consisting of props and state) into rich HTML.

Design_be_compact_to_be_elegant

First, let’s understand the process of re-rendering in React. A React component translates raw data (consisting of props and state) into rich HTML. You can control the state of the component by changing its props and state. The entire component is re-rendered whenever its props/state change.


Steps for re-rendering in React:


  1. When the props/state of a component change, React builds a new virtual DOM and compares the new and old virtual DOMs using the diff algorithm.
  2. If the new and old virtual DOM trees are not consistent, re-rendering begins.


1

The DOM operation is very time-consuming. To improve the performance of components, try to minimize unnecessary re-rendering of them.


Consider the following example.


The following figure shows a rendered component.


2

As the state of the component changes, the green node is re-rendered.

3

It is logical to think that only three nodes (highlighted in green) need to be updated to complete the component change.

4

However, as per React’s updating rules, whenever the props or state of a component change, the entire component is re-rendered. So, apart from the three green nodes, all other nodes (highlighted in yellow) are also re-rendered.


5

This is a huge waste of performance. For complex pages, this results in a very sub-standard user-experience. Therefore, to improve component performance, it is imperative to minimize unnecessary rendering.

Component Optimization Techniques


Due to the need for minimizing unnecessary rendering, let’s look at some techniques for component optimization.

Pure Component


A pure component in React is a component whose render function is dependent only on its props and state. With the same props and state, the rendered output receives the same results. The following code represents a pure component.

render() {
     return (
          {this.state.rows}   
     );
}

shouldComponentUpdate function


The function, shouldComponentUpdate, runs before rendering. Its return value determines whether the component is re-rendered.



  • Return value = True

    Whenever the component’s props/state change, the virtual DOM is re-constructed and compared using the diff algorithm. The comparison result decides whether the entire component should be re-rendered.


  • Return value = False

    The component is not re-rendered.


The default return value of the function is True, so re-rendering may occur. When re-rendering is not required, the default return value of the shouldComponentUpdate function is set to False.


The position of the shouldComponentUpdate function during component re-rendering is as shown in the following figure.

6

PureRenderMixin

React provides the official PureRenderMixin plugin, which makes the shouldComponentUpdate function return False. The plugin can reduce unnecessary re-rendering and improve performance to a certain extent. The use of the plugin is as follows:

 
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 <div className={this.props.className}>foo</div>;
  }
}

From the source code, you can see that the plugin is overwriting the shouldComponentUpdate function.

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

This method makes a brief comparison between the current state and next state of a component. If the component’s state changes, a False result is returned. Otherwise, the True result is returned. This function can be further decomposed for implementation.

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

It compares the changes of props/state of the component. In the latest version of React, the basic class of React.PureComponent is provided, and the PureRenderMixin plugin is no longer needed.

Status Comparison

Let’s assume every component uses the PureRenderMixin plugin. Consider the following component.

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

To change the color of the component to blue, the following code is run.

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

}

The state comparison between two simple objects is depicted as:

7

If the two objects compared are complex, such as dates, functions, or objects with multiple nested layers, then the comparisons are either very time-consuming or unavailable.

You can overwrite the shouldComponentUpdate function to make it applicable to any item (including deep comparison/comparison of recursive layers). Deep comparison is time-consuming and hence not recommended. You should try to use simple props and state as well as avoid unnecessary attributes (or attributes calculated by other attributes in the state).

Immutable.js

The comparison of complex data is time-consuming or unavailable in React. But Immutable.js can solve this problem. It returns the same reference for immutable objects and returns a new reference for changed objects. The following code is used for comparison of the state.

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

Dynamic/Static Separation


Consider a component, such as a table.


<ScrollTable
width={300}

color='blue'

scrollTop={this.props.offsetTop}
/>

This is a scrollable table. The offsetTop represents the distance of visible area from the upper border of a browser. When the mouse scrolls, the value changes, altering the props of the component. Subsequently, the component is re-rendered.
Now, consider the following code.

<OuterScroll>
&lt;InnerTable width={300} color='blue'/&gt;
</OuterScroll>

The props of the InnerTable component is fixed. So, use of the pureRenderMixin plugin ensures the return value of shouldComponentUpdate as False. Irrespective of the state of OuterScroll and parent component, the InnerTable component is not to be re-rendered. In short, the sub-component isolates the state of the parent component.


By separating the changed and unchanged attributes, re-rendering is reduced, and performance improves. At the same time, the components can be easily separated and reused.

Child Components


In nested multi-layer and complex components, there are many sub-nodes. So, component updates take longer, making the component difficult to maintain. The unidirectional flow of information from parent component to child component may lead to loss of control over the components.

  • Children change over time

    Consider the following component, which is re-rendered every second (please note: this is a theoretical example only and not feasible in React).

class Parent extends Component {
  shouldComponentUpdate(nextProps) {
      return this.props.children != nextProps.children;
  }
  render() {
      return &lt;div&gt;{this.props.children}&lt;/div&gt;;
  }
}
setInterval(() => {
   ReactDOM.render(

      &lt;Parent&gt;

          &lt;div&gt;child&lt;/div&gt;

      &lt;/Parent&gt;

  );
}, 1000);

Children components that need to be re-rendered are recognized using the shouldComponentUpdate function by checking if the children and parent components are identical or not. Children is an attribute of props, so it is same all of the time. In theory, the component is not to be re-rendered. But in practice, it is re-rendered every time.


Let’s look at the structure of children in the following code.


8

Children are relatively complex objects and re-constructed during each component update. In other words, children are dynamically constructed, so the update is not equal every time. As a result, shouldComponentUpdate returns True every time, and the component is re-rendered. We can replace children with a variable in order for the same object to be constructed every time.

Independent children


Consider the following component.

 class TwoColumnSplit extends Component {
      shouldComponentUpdate() {
          return false;
      }
      render() {
          return (
       <div>
                <FloatLeft>{this.props.children[0]}</FloatLeft>
                <FloatRight>{this.props.children[1]}</FloatRight>
            </div>
          );
      }
  } <TwoColumnSplit>
      <TargetContainer/>
      <BudgetContainer/>
   </TwoColumnSplit>

The shouldComponentUpdate function returns False, and the component doesn’t change with outside state changes. This is because the components TargetContainer and BudgetContainer did not get any information from their parent element, and the children and the parent component are isolated. In fact, TwoColumnSplit plays the role of isolation. For components that do not need to get data from the outside, you can isolate them from external changes by returning a False value to reduce re-rendering.

Container and Component


You can also isolate external changes by using component containers. A container is a data layer, with the component responsible for rendering corresponding components based on data obtained, without any data interaction. The following shows a container and its components.

class BudgetContainer extends Component {
  constructor(props) {
      super(props);
      this.shouldComponentUpdate = PureRenderMixin.shouldComponentUpdate.bind(this);
  }
  computeState() {
      return BudgetStore.getStore()
  }
  render() {
return <Budget {...this.state}/>
   }
}

Containers should not have props and children, so a container and its parent component are isolated. Therefore, no re-rendering is needed due to external factors.


To move the position of a component, you only have to put the component in the right position. You can move it to any place, as well as use in different applications and tests. You should consider the source of internal data to write different containers in different environments.

Conclusion


We covered the importance of re-rendering efficiently in React. Also, you learned various techniques for component optimization, including shouldComponentUpdate and PureRenderMixin functions, status comparison, Immutable.js, dynamic/static separation, child components, and containers.

目录
相关文章
|
4月前
|
前端开发 搜索推荐 API
【Prompt Engineering:ReAct 框架】
ReAct 框架由 Yao 等人(2022)提出,结合大语言模型(LLMs)生成推理轨迹与任务操作,交替进行推理与行动。此框架允许模型与外部环境(如知识库)互动,以动态更新操作计划并处理异常。ReAct 在语言和决策任务上表现优异,提升模型的人类可解释性和可信度。研究显示,ReAct 优于多个基准模型,尤其在结合链式思考时效果最佳。通过实例演示,ReAct 能有效整合内外部信息,优化推理过程。
198 9
【Prompt Engineering:ReAct 框架】
|
8月前
|
存储 程序员 编译器
Modern C++
Modern C++
|
前端开发 API
react报错ReactDOM.render is no longer supported in React 18. 解决
react报错ReactDOM.render is no longer supported in React 18. 解决
300 0
|
前端开发 数据安全/隐私保护
React高阶组件(Higher-Order Components)及其应用
React高阶组件(Higher-Order Components)及其应用
95 0
|
算法
antd-design-pro使用操作
antd-design-pro使用操作
247 0
|
前端开发
react项目实战学习笔记-学习9-ReactDOM.render is no longer supported in React 18
react项目实战学习笔记-学习9-ReactDOM.render is no longer supported in React 18
155 0
|
Web App开发 JavaScript 前端开发
Modern模式引发qiankun的一场“命案”
前沿:文章的起源在于开发环境中使用qiankun框架,父应用加载子应用遇到一则报错 Failed to load module script: The server responded with a non-JavaScript MIME type of "text/html". Strict MIME type checking is enforced for module scripts per HTML spec 直接的翻译意思就是加载模块脚本失败:服务器以非JavaScript MIME类型“text/html”去响应。而也是这个问题引发了我的思考,到底是什么问题导致?
1006 0
Modern模式引发qiankun的一场“命案”
|
前端开发 JavaScript
Fragments(精读React官方文档—18)
Fragments(精读React官方文档—18)
107 0
|
JavaScript 前端开发
svelte教程(8)stores
有时,您将需要多个不相关的组件或常规的JavaScript模块访问这些值。 在Svelte,我们通过store来做到这一点。store只是一种对象,该对象具有一种subscribe方法,该方法允许在store的value发生变化时通知订阅过的组件。
3400 0
|
JavaScript 前端开发 Python

热门文章

最新文章

下一篇
开通oss服务