React中的domDiffing算法

1.react/vue中的key有什么作用?(key的內(nèi)部原理是什么)

(1)簡(jiǎn)單地說:key是虛擬DOM對(duì)象的標(biāo)識(shí)录粱,在更新顯示時(shí)key起著極其重要的作用

(2)詳細(xì)的說:當(dāng)狀態(tài)中的數(shù)據(jù)發(fā)生變化時(shí)杏死,react會(huì)根據(jù)【新數(shù)據(jù)】生成【新的虛擬DOM】摹芙,隨后React進(jìn)行【新虛擬DOM】與【舊虛擬DOM】的diff比較嵌莉,比較規(guī)則如下:

  • 舊虛擬DOM找到了與新虛擬DOM相同的key:
    若虛擬DOM中內(nèi)容沒變苛茂,直接使用之前的真實(shí)DOM
    若虛擬DOM中內(nèi)容變了之宿,則生成新的真實(shí)DOM族操,隨后替換掉頁(yè)面中之前的真實(shí)DOM
  • 舊的虛擬DOM中未找到與新虛擬DOM相同的key
    根據(jù)數(shù)據(jù)創(chuàng)建新的真實(shí)DOM,隨后渲染到頁(yè)面

2.為什么遍歷列表時(shí),key最好不要使用index

用index作為key可能引發(fā)的問題:

(1)若對(duì)數(shù)據(jù)進(jìn)行:逆序添加色难,逆序刪除等破壞順序操作
會(huì)產(chǎn)生沒有必要的真實(shí)DOM更新 ===> 界面效果沒問題泼舱,但是效率低

(2)如果結(jié)構(gòu)中還包括輸入類的DOM:
會(huì)產(chǎn)生錯(cuò)誤的DOM更新 ===> 界面有問題

(3)注意!如果不存在對(duì)數(shù)據(jù)的逆序添加枷莉,逆序刪除等破壞順序操作娇昙,
僅用于渲染列表用于展示,使用index作為key是沒有問題的

例:

class Person extends React.Component {
  state = {
    persons: [
      { id: 1, name: "小張", age: 18 },
      { id: 2, name: "小李", age: 19 }
    ]
  };

  add=()=>{
    const {persons}=this.state;
    const p={id:persons.length+1,name:"小王",age:20};
    this.setState({persons:[p,...persons]});
  };

  render() {
    return (
      <div>
        <h2>展示人員信息</h2>
        <h3>使用index(索引值)作為key</h3>
        <button onClick={this.add}>添加一個(gè)小王</button>
        <ul>
          {
            this.state.persons.map((personObj, index) => {
              return <li key={index}>
                        {personObj.name}---{personObj.age}
                        <input type="text"/>
                     </li>
            })
          }
        </ul>
        <hr/><hr/>
        <h3>使用id(數(shù)據(jù)唯一標(biāo)識(shí))作為key</h3>
        <ul>
          {
            this.state.persons.map((personObj) => {
              return <li key={personObj.id}>
                        {personObj.name}---{personObj.age}
                        <input type="text"/>
                      </li>
            })
          }
        </ul>
      </div>
    )
  }
}
  
ReactDOM.render(<Person />, document.getElementById("test"));
  • 使用index索引作為key的程序執(zhí)行流程:

    初始數(shù)據(jù)

    { id: 1, name: "小張", age: 18 },
    { id: 2, name: "小李", age: 19 }
    

    初始虛擬DOM

    <li key=0>小張---18<input type="text"/></li>
    <li key=1>小李---19<input type="text"/></li>
    

    更新后的數(shù)據(jù)

    { id: 3, name: "小王", age: 20 },
    { id: 1, name: "小張", age: 18 },
    { id: 2, name: "小李", age: 19 }
    

    更新后的虛擬DOM

    <li key=0>小王---20<input type="text"/></li> 
          --->生成新的虛擬DOM,更新li中的內(nèi)容笤妙,但是復(fù)用了input
    <li key=1>小張---18<input type="text"/></li> 
          --->生成新的虛擬DOM,更新li中的內(nèi)容冒掌,但是復(fù)用了input
    <li key=2>小李---19<input type="text"/></li> 
          --->生成新的虛擬DOM,更新li中的內(nèi)容,并且添加了新的input
    

    如果使用index作為key的值蹲盘,且新添加的內(nèi)容在數(shù)組的前面股毫,則每次的更新都不會(huì)復(fù)用上一次的虛擬DOM,增加程序開銷

  • 使用id唯一標(biāo)識(shí)作為key

    初始數(shù)據(jù)

    { id: 1, name: "小張", age: 18 },
    { id: 2, name: "小李", age: 19 }
    

    初始虛擬DOM

    <li key=1>小張---18<input type="text"/></li>
    <li key=2>小李---19<input type="text"/></li>
    

    更新后的數(shù)據(jù)

    { id: 3, name: "小王", age: 20 },
    { id: 1, name: "小張", age: 18 },
    { id: 2, name: "小李", age: 19 }
    

    更新后的虛擬DOM

    <li key=3>小王---20<input type="text"/></li> 
          --->生成新的虛擬DOM,更新li中的內(nèi)容召衔,并且添加了新的input
    <li key=1>小張---18<input type="text"/></li> 
          --->復(fù)用原來的li中的內(nèi)容及input
    <li key=2>小李---19<input type="text"/></li> 
          --->復(fù)用原來的li中的內(nèi)容及input
    

3.開發(fā)中如何選擇key?

(1)最好使用每條數(shù)據(jù)的唯一標(biāo)識(shí)作為key铃诬,比如id、手機(jī)號(hào)苍凛、身份證號(hào)趣席、學(xué)號(hào)等唯一值
(2)如果確定只是簡(jiǎn)單的展示數(shù)據(jù),用index也是可以的

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末醇蝴,一起剝皮案震驚了整個(gè)濱河市宣肚,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌哑蔫,老刑警劉巖钉寝,帶你破解...
    沈念sama閱讀 217,542評(píng)論 6 504
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異闸迷,居然都是意外死亡嵌纲,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,822評(píng)論 3 394
  • 文/潘曉璐 我一進(jìn)店門腥沽,熙熙樓的掌柜王于貴愁眉苦臉地迎上來逮走,“玉大人,你說我怎么就攤上這事今阳∈Γ” “怎么了?”我有些...
    開封第一講書人閱讀 163,912評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵盾舌,是天一觀的道長(zhǎng)墓臭。 經(jīng)常有香客問我,道長(zhǎng)妖谴,這世上最難降的妖魔是什么窿锉? 我笑而不...
    開封第一講書人閱讀 58,449評(píng)論 1 293
  • 正文 為了忘掉前任酌摇,我火速辦了婚禮,結(jié)果婚禮上嗡载,老公的妹妹穿的比我還像新娘窑多。我一直安慰自己,他們只是感情好洼滚,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,500評(píng)論 6 392
  • 文/花漫 我一把揭開白布埂息。 她就那樣靜靜地躺著,像睡著了一般遥巴。 火紅的嫁衣襯著肌膚如雪千康。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,370評(píng)論 1 302
  • 那天挪哄,我揣著相機(jī)與錄音吧秕,去河邊找鬼。 笑死迹炼,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的颠毙。 我是一名探鬼主播斯入,決...
    沈念sama閱讀 40,193評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼蛀蜜!你這毒婦竟也來了刻两?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,074評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤滴某,失蹤者是張志新(化名)和其女友劉穎磅摹,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體霎奢,經(jīng)...
    沈念sama閱讀 45,505評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡户誓,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,722評(píng)論 3 335
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了幕侠。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片帝美。...
    茶點(diǎn)故事閱讀 39,841評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖晤硕,靈堂內(nèi)的尸體忽然破棺而出悼潭,到底是詐尸還是另有隱情,我是刑警寧澤舞箍,帶...
    沈念sama閱讀 35,569評(píng)論 5 345
  • 正文 年R本政府宣布舰褪,位于F島的核電站,受9級(jí)特大地震影響疏橄,放射性物質(zhì)發(fā)生泄漏占拍。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,168評(píng)論 3 328
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望刷喜。 院中可真熱鬧残制,春花似錦、人聲如沸掖疮。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,783評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)浊闪。三九已至恼布,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間搁宾,已是汗流浹背折汞。 一陣腳步聲響...
    開封第一講書人閱讀 32,918評(píng)論 1 269
  • 我被黑心中介騙來泰國(guó)打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留盖腿,地道東北人爽待。 一個(gè)月前我還...
    沈念sama閱讀 47,962評(píng)論 2 370
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像翩腐,于是被迫代替她去往敵國(guó)和親鸟款。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,781評(píng)論 2 354

推薦閱讀更多精彩內(nèi)容

  • 1. React簡(jiǎn)介 React 起源于 Facebook 的內(nèi)部項(xiàng)目茂卦,因?yàn)樵摴緦?duì)市場(chǎng)上所有 JavaScrip...
    王蕾_fd49閱讀 409評(píng)論 0 0
  • 1. javascript的typeof返回哪些數(shù)據(jù)類型. 答案:string,boolean,number,un...
  • Vue -漸進(jìn)式JavaScript框架 介紹 vue 中文網(wǎng) vue github Vue.js 是一套構(gòu)建用戶...
    桂_3d6b閱讀 821評(píng)論 0 0
  • jquery介紹 jQuery是目前使用最廣泛的javascript函數(shù)庫(kù) 據(jù)統(tǒng)計(jì)何什,全世界排名前100萬的網(wǎng)站,有...
    就是這么帥_567e閱讀 1,143評(píng)論 0 0
  • 0 HTML5相關(guān) websocket WebSocket 使用ws或wss協(xié)議等龙,Websocket是一個(gè)持久化的...
    可愛多小姐閱讀 869評(píng)論 0 0