Understanding the virtual DOM? What does the virtual DOM mainly do? What is the virtual DOM itself?
Essentially, the Virtual Dom is a JavaScript object that represents the Dom structure as an object. Abstract the page state into the form of JS objects, with different rendering tools, so that cross-platform rendering is possible. Through the transaction processing mechanism, the results of multiple DOM modifications are updated to the page at one time, thus effectively reducing the number of page rendering, reducing the number of DOM modification redrawing and rearrangement, and improving rendering performance.
The virtual DOM is an abstraction of the DOM, and this object is a more lightweight representation of the DOM. For example, Node.js does not have DOM. If you want to implement SSR, then one way is to use the virtual DOM, because the virtual DOM itself is a JS object. Vue or React converts the code into an object (the virtual DOM) before rendering it to the page. The actual DOM structure is described as an object and eventually rendered to a page. The virtual DOM caches a copy of the data each time it changes, and the current virtual DOM is compared to the cached one when it changes. The DIff algorithm is encapsulated inside VUE or React to compare, modify the changed changes during rendering, and render the original data that did not change.
Is also a basic requirement of the modern front frame without manual DOM, on the one hand because of manual DOM cannot assure program performance, collaboration projects if you don’t strict review, there may be a developer write performance low code, on the other hand, more important is to omit the manual DOM manipulation can greatly improve the development efficiency.
Why use the Virtual DOM:
(1) Ensure the performance lower limit, in the case of no manual optimization, to provide acceptable performance
Let’s compare the real AND Virtual DOM operations when modifying the DOM to see the performance cost of their rearrangement and redrawing:
- Real DOM: Generate HTML string + rebuild all DOM elements
- Virtual DOM: Generates vNode+ DOMDiff+ necessary DOM updates
Virtual DOM update DOM preparation takes more time, that is, at the JS level, which is extremely cheap compared to more DOM manipulation. The framework gives you the assurance that you don’t need to manually optimize and I can still give you decent performance, Yu said in a community forum.
(2) Cross-platform Virtual DOM is essentially an object of JavaScript, which can be operated conveniently across platforms, such as server-side rendering and UNIApp
How does the React Diff algorithm work?
In fact, diff algorithm is to explore the way of generating DOM tree update patches after the virtual DOM tree changes. It compares the changes of the old and new virtual DOM trees and applies the update patch to the real DOM to complete the view update at the lowest cost.The specific process is as follows:
- The real DOM is first mapped to the virtual DOM;
- When the virtual DOM changes, the patch will be calculated and generated according to the gap. The patch is a structured data, including adding, updating and removing, etc.
- Update the real DOM according to patch and feed back to the user interface.
A simple example:
import React from 'react' export default class ExampleComponent extends React.Component { render() { if(this.props.isVisible) { return <div className="visible">visbile</div>; } return <div className="hidden">hidden</div>; }}Copy the code
Here, you first assume that ExampleComponent is visible, and then change its state to make it invisible. For a real DOM operation, React creates a div node.
<div class="visible">visbile</div>
Copy the code
When the visbile value is changed to false, the class property is replaced with hidden and the innerText is overwritten to hidden. This process of generating patches and updating differences is called diff algorithm.
Diff algorithm can be summarized as three strategies to optimize complexity from tree, component and element levels:
Strategy 1: Ignore the cross-layer operation of nodes to improve comparison efficiency. (Tree based comparison)
This strategy requires a tree comparison, that is, a hierarchical comparison of trees. The method of tree comparison is very “violent”, that is, two trees only compare nodes at the same level. If a node is found to be no longer existing, the node and its child nodes will be completely deleted and will not be used for further comparison, which improves the comparison efficiency.
Strategy 2: If the classes of components are consistent, the default is similar tree structure; otherwise, the default is different tree structure. (Component based comparison)
During component alignment:
- If the components are of the same type, tree alignment is performed;
- If not, add them to the patch.
As long as the parent component type is different, it will be re-rendered. This is why shouldComponentUpdate, PureComponent, and React. Memo can improve performance.
Strategy 3: You can compare child nodes at the same level by marking keys. (Comparison based on nodes)
Element alignment occurs primarily at the same level and patches are generated by marking node operations. Node operations include insert, move, and delete. Among them, node reordering involves insertion, movement and deletion at the same time, so the efficiency consumption is the largest. In this case, strategy 3 plays a crucial role. React can move DOM nodes directly by marking keys, reducing internal friction
Why does the React key work? Key is mainly used to solve what kind of problems
Keys are an auxiliary identifier used by React to track which elements in a list were changed, added, or removed. During development, we need to ensure that an element’s key is unique among its siblings.
In the React Diff algorithm, React uses the Key to determine whether an element is newly created or moved, thus reducing unnecessary element re-rendering. In addition, React also uses the Key to determine the relationship between an element and its local state.
Matters needing attention:
- The key value must correspond to the specific element – 1;
- Try not to use array index as key;
- Don’t use random numbers or other operations to render elements with unstable keys, which can cause worse performance overhead than without keys.
Which is more efficient than the introduction of the virtual DOM versus manipulating the native DOM directly, and why
The virtual DOM is not necessarily more efficient than the native DOM, and if you only modify the text of a button, the virtual DOM is unlikely to operate faster than the real DOM in any case. When rendering a large number of DOM for the first time, the virtual DOM is also slower than innerHTML insertion due to the extra layer of virtual DOM computation. It provides a performance floor and is faster when it comes to optimizing for real DOM operations. So according to the specific scene to discuss.
Throughout the evolution of DOM manipulation, the main contradiction was not performance, but how well the developer wrote, the development experience/development efficiency. The virtual DOM is nothing more than a high-order product created by front-end developers in pursuit of better r&d experience and efficiency. The Virtual DOM doesn’t necessarily lead to better performance, and React has never officially advertised the virtual DOM as a performance selling point. The advantage of the virtual DOM is that it can provide a more streamlined and efficient mode of development (i.e., functional UI programming) while still maintaining decent performance.
How does React differ from Vue’s diff algorithm?
Diff algorithm is a method to generate update patches, which is mainly used to update the real DOM after the virtual DOM tree changes. Therefore, the DIff algorithm must have such a process: triggering updates → generating patches → applying patches.
The React diff algorithm triggers updates mainly after state changes and hooks calls. At this point, the virtual DOM tree change traversal is triggered and the depth-first traversal algorithm is adopted. However, the traditional traversal method is less efficient. To optimize efficiency, divide-and-conquer was used. The single node comparison is transformed into three types of node comparison, namely tree, component and element, to improve efficiency.
- Tree comparison: Since there is less movement between nodes in the web view, the two virtual DOM trees only compare nodes at the same level.
- Component alignment: If components are of the same type, tree alignment is performed. If components are not of the same type, they are directly added to the patch.
- Element alignment: This occurs mainly at the same level, and patches are generated by marking node operations that correspond to real DOM clipping operations.
This is the classic React Diff algorithm. React 16 introduced Fiber architecture. In order to make the whole update process can be suspended and resumed at any time, FiberNode and FiberTree were used for node and tree reconstruction respectively. FiberNode uses a double-linked list structure to find siblings and children directly. The whole update process was completed by double buffering of current and workInProgress trees. After workInProgress is updated, modify the current pointer to point to the new node.
Vue’s overall DIff strategy is aligned with React. Although it lacks time slicing capability, this does not mean that Vue performance is worse because it was introduced in Vue 3 at the beginning and removed later due to low returns. With the exception of high frame rate animations, almost all scenes in Vue can be improved by using stabilization and throttling.