React Reconciliation

React给我们提供了很多的API,比如setState或者hook(setXXX)帮助我们更新state,从而在页面上展示新的数据。

我们只知道调用了更新state的API,react会默默的帮助我们去做UI的update,但是并不知道其中的更新的过程。接下来我会给大家介绍state改变之后,react帮助我们做的事情。

react UI update流程

  • 组件state/props发生改变
  • 触发组件render函数的执行,render函数会创建一个新的react element tree(js object/aka virtual dom)
  • react会根据某一种diff算法,对比新产生的react element tree和之前tree之间,有哪些element需要被更新
  • 更新UI(真实DOM)

Reconciliation

react Reconciliation 算法就是所谓的react diff算法,这种算法用来compare新旧的react element tree(virtual dom),找到一个更新真实UI的最高效的方式。

算法的大概思路是:
react会从上往下diff react element tree,也就是从root element开始向下diff。

image.png

接下来的diff方式取决于root element的类型:

React DOM Element

当root element是一个普通的HTML tag,比如div、span等标签

Same type

如果新旧的virtual DOM tree的某一个root节点dom类型完全相同,那么react会去check DOM元素的所有属性,最终只会update改变的DOM属性,而不会updateDOM元素。

different type

如果新旧DOM tree的root element dom类型不同,那么react会直接将当前的这个root element以及其tree上的所有节点全部删除,创建新的root element,并重新构建其树中的所有其他元素。

example:

image.png

当react check到新的树的某一个root element的类型不同,那么react会直接删除div,unmount Counter组件,然后创建新的span,并且构建新的ounter instance。

那么组件的lifecycle的执行顺序是:

  • 老Counter组件的componentWillUnmount调用
  • 新Counter组件的constructor调用
  • 新Counter组件的getDerivedStateFromProps调用
  • 新Counter组件的第一次render调用
  • 新Counter组件的componentDidMount调用

React Component Element

Same class/function Component

如果新旧DOM Tree的root component的类型完全相同,那么react只会更新当前的component element的instance,让当前的root component进入Updating阶段

example:

// old
<Counter number=1 />

// new
<Counter number=2 />

React check新旧component element都是Counter,这时候当前的counter组件的生命周期方法会按照顺序触发

  • getDerivedStateFromProps()
  • shouldComponentUpdate()
  • render()
  • getSnapshotBeforeUpdate()
  • componentDidUpdate()

Diff算法处理列表

react处理列表并没有什么特殊,但是对于不同的情况,可能和你想象的顺序不同,比如对于下面这些情况:

在列表的尾部插入新的元素

// old

<ul>
    <li>1</li>
    <li>2</li>
</ul>

<ul>
    <li>1</li>
    <li>2</li>
    <li>3</li>
</ul>

react首先check root element ul,发现type和属性均没有改变,因此依次diff所有子节点<li>,发现有新的子节点<li>3</li>,就会创建

在列表的头部插入新的元素

// old

<ul>
    <li>1</li>
    <li>2</li>
</ul>

<ul>
    <li>3</li>
    <li>1</li>
    <li>2</li>
</ul>

react首先check root element ul,发现type和属性均没有改变,因此依次diff所有子节点<li>。
diff第一个li,发现其子元素text发生了改变1 ---> 3,因此之后要update这个元素

diff第二个li,发现其子元素text发生了改变2 ---> 1,因此之后要update这个元素

由于第三个li,以前并不存在,因此创建新的元素<li>2</li>

这流程似乎和我们想象的直接创建新的<li>3</li>不一样,因此如果在列表的头部插入元素,可能会导致performance变差。

因此为了解决此类列表问题,react引入了key属性。

使用key

对于像这种list这样的组件,他们通常都使用一样的DOM type以及类似的属性和结构。

react在没有特别的设置下,会按照顺序从左向右依次diff list中的每一个元素,因此对于list头部插入元素的情况,会导致list中的每一个元素都被更新。

为了提高list的diff效率,react期待我们给每一个list item都加上一个id,也就是key,让react在diff中这些非常相似的item时,尝试按照key去diff。

也就是key相同的元素,react就默认这个元素就是之前的那个元素,只需要check是否有改变的属性,只进行Update。对于新的key值,直接创建新的元素。

example

// old

<ul>
    <li key='1'>1</li>
    <li key='2'>2</li>
</ul>

// new 
<ul>
    <li key='3'>3</li>
    <li key='1'>1</li>
    <li key='2'>2</li>
</ul>

对于上面这种情况,react发现key='1'以及key='2'元素完全没变属性也没变,因此不做任何update,而只是创建一个新的<li key='3'>3</li>

为什么最好不要使用array的index作为key值?
  • case1

在原数组的头部加入一个新的元素

// old

<ul>
    {[1,2].map((value,index) => <li key={index}>{value}</li>)}
</ul>

// new 


<ul>
    {[3,1,2].map((value,index) => <li key={index}>{value}</li>)}
</ul>

如果原数组的头部加了3,那么diff流程和不加key完全一样,所有元素都需要被update,并且创建新的元素<li>2</li>

  • case2

从原数组的中间删除一个元素。

// old

<ul>
    {[1,2,3].map((value,index) => {
        return (
            <div>
                <label>{value}</label>
                <input />
            </div>
        )
    })}
</ul>

// new 


<ul>
    {[1,3].map((value,index) => {
        return (
            <div>
                <label>{value}</label>
                <input />
            </div>
        )
    })}
</ul>

对于这种情况,react diff的流程:

  • diff第一个元素1没问题,属性不变不更新。
  • diff第二个元素3的时候,发现他的key是1,react惊喜的发现以前就一个key是1的元素,那么就把以前的2元素更新成3吧,其他部分比如input部分没有任何属性改变,那就不更新吧。
  • 发现新树不再存在key是2的元素,于是将以前的3元素直接删除

你本来期待的是让react把中间的元素删除,但是react只是在原来的基础上更新了第二个元素,而删除了第三个元素。这时候会造成,其他本来和元素2配套的组件(比如input),现在变成了和元素3配套,本应该和元素3配套的组件被删除了。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,362评论 5 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,330评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,247评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,560评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,580评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,569评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,929评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,587评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,840评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,596评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,678评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,366评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,945评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,929评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,165评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,271评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,403评论 2 342

推荐阅读更多精彩内容