react的diff方法可用于找出两个对象之间的差异,目的是尽可能做到节点复用;diff算法是调和的具体实现,而调和是指将virtual dom树转换成actual dom树的最少操作的过程。
本教程操作环境:windows10系统、react18.0.0版、dell g3电脑。
react的diff方法是什么?
一、diff算法的作用
渲染真实dom的开销很大,有时候我们修改了某个数据,直接渲染到真实dom上会引起整个dom树的重绘和重排。我们希望只更新我们修改的那一小块dom,而不是整个dom,diff算法就帮我们实现了这点。
diff算法的本质就是:找出两个对象之间的差异,目的是尽可能做到节点复用。
注:此处说到的对象,指的其实就是vue中的virtual dom(虚拟dom树),即使用js对象来表示页面中的dom结构。
二、react的diff算法 1、什么是调和? 将virtual dom树转换成actual dom树的最少操作的过程称为调和。
2、什么是react diff算法? diff算法是调和的具体实现。
3、diff策略 react用三大策略 将o(n3)复杂度 转化为o(n)复杂度
(1)策略一(tree diff):web ui中dom节点跨层级的移动操作特别少,可以忽略不计。
(2)策略二(component diff):拥有相同类的两个组件 生成相似的树形结构,拥有不同类的两个组件 生成不同的树形结构。
(3)策略三(element diff):对于同一层级的一组子节点,通过唯一id区分。
4、tree diff:(1)react通过updatedepth对virtual dom树进行层级控制。
(2)对树分层比较,两棵树只对同一层次节点进行比较。如果该节点不存在时,则该节点及其子节点会被完全删除,不会再进一步比较。
(3)只需遍历一次,就能完成整棵dom树的比较。
如果dom 节点出现了跨层级操作,diff会怎么办?
答:tree diff是对树的每一层进行遍历,如果某组件不存在了,则会直接销毁。如图所示,左边是旧属,右边是新属,第一层是r组件,一模一样,不会发生变化;第二层进入component diff,同一类型组件继续比较下去,发现a组件没有,所以直接删掉a、b、c组件;继续第三层,重新创建a、b、c组件。
如上图所示,以a为根节点的整棵树会被重新创建,而不是移动,因此 官方建议不要进行dom节点跨层级操作,可以通过css隐藏、显示节点,而不是真正地移除、添加dom节点。
5、component diff :react对不同的组件间的比较,有三种策略
(1)同一类型的两个组件,按原策略(层级比较)继续比较virtual dom树即可。
(2)同一类型的两个组件,组件a变化为组件b时,可能virtual dom没有任何变化,如果知道这点(变换的过程中,virtual dom没有改变),可节省大量计算时间,所以用户可以通过 shouldcomponentupdate() 来判断是否需要判断计算。
(3)不同类型的组件,将一个(将被改变的)组件判断为dirtycomponent(脏组件),从而替换整个组件的所有节点。
注意:如上图所示,当组件d变为组件g时,即使这两个组件结构相似,一旦react判断d和g是不用类型的组件,就不会比较两者的结构,而是直接删除组件d,重新创建组件g及其子节点。虽然当两个组件是不同类型但结构相似时,进行diff算法分析会影响性能,但是毕竟不同类型的组件存在相似dom树的情况在实际开发过程中很少出现,因此这种极端因素很难在实际开发过程中造成重大影响。
6、element diff 当节点处于同一层级时,diff提供三种节点操作:删除、插入、移动。
插入:组件 c 不在集合(a,b)中,需要插入
删除:
(1)组件 d 在集合(a,b,d)中,但 d的节点已经更改,不能复用和更新,所以需要删除 旧的d ,再创建新的。
(2)组件d之前在集合(a,b,d)中,但集合变成新的集合(a,b)了,d 就需要被删除。
移动:组件d已经在集合(a,b,c,d)里了,且集合更新时,d没有发生更新,只是位置改变,如新集合(a,d,b,c),d在第二个,无须像传统diff,让旧集合的第二个b和新集合的第二个d 比较,并且删除第二个位置的b,再在第二个位置插入d,而是 (对同一层级的同组子节点) 添加唯一key进行区分,移动即可。
移动情形一:新旧集合中存在相同节点但位置不同时,如何移动节点
(1)b不移动,不赘述,更新l astindex=1
(2)新集合取得 e,发现旧不存在,故在lastindex=1的位置 创建e,更新lastindex=1
(3)新集合取得c,c不移动,更新lastindex=2
(4)新集合取得a,a移动,同上,更新lastindex=2
(5)新集合对比后,再对旧集合遍历。判断 新集合 没有,但 旧集合 有的元素(如d,新集合没有,旧集合有),发现 d,删除d,diff操作结束。
react中diff算法实现的代码:
_updatechildren: function(nextnestedchildrenelements, transaction, context) { var prevchildren = this._renderedchildren; var removednodes = {}; var mountimages = []; // 获取新的子元素数组 var nextchildren = this._reconcilerupdatechildren( prevchildren, nextnestedchildrenelements, mountimages, removednodes, transaction, context ); if (!nextchildren && !prevchildren) { return; } var updates = null; var name; var nextindex = 0; var lastindex = 0; var nextmountindex = 0; var lastplacednode = null; for (name in nextchildren) { if (!nextchildren.hasownproperty(name)) { continue; } var prevchild = prevchildren && prevchildren[name]; var nextchild = nextchildren[name]; if (prevchild === nextchild) { // 同一个引用,说明是使用的同一个component,所以我们需要做移动的操作 // 移动已有的子节点 // notice:这里根据nextindex, lastindex决定是否移动 updates = enqueue( updates, this.movechild(prevchild, lastplacednode, nextindex, lastindex) ); // 更新lastindex lastindex = math.max(prevchild._mountindex, lastindex); // 更新component的.mountindex属性 prevchild._mountindex = nextindex; } else { if (prevchild) { // 更新lastindex lastindex = math.max(prevchild._mountindex, lastindex); } // 添加新的子节点在指定的位置上 updates = enqueue( updates, this._mountchildatindex( nextchild, mountimages[nextmountindex], lastplacednode, nextindex, transaction, context ) ); nextmountindex++; } // 更新nextindex nextindex++; lastplacednode = reactreconciler.gethostnode(nextchild); } // 移除掉不存在的旧子节点,和旧子节点和新子节点不同的旧子节点 for (name in removednodes) { if (removednodes.hasownproperty(name)) { updates = enqueue( updates, this._unmountchild(prevchildren[name], removednodes[name]) ); } } }
三、基于diff的开发建议基于tree diff:
开发组件时,注意保持dom结构的稳定;即,尽可能少地动态操作dom结构,尤其是移动操作。当节点数过大或者页面更新次数过多时,页面卡顿的现象会比较明显。这时可以通过 css 隐藏或显示节点,而不是真的移除或添加 dom 节点。基于component diff:
注意使用 shouldcomponentupdate() 来减少组件不必要的更新。对于类似的结构应该尽量封装成组件,既减少代码量,又能减少component diff的性能消耗。基于element diff:
对于列表结构,尽量减少类似将最后一个节点移动到列表首部的操作,当节点数量过大或更新操作过于频繁时,在一定程度上会影响 react 的渲染性能。推荐学习:《react视频教程》
以上就是react的diff方法是什么的详细内容。