React應(yīng)用架構(gòu)

【插播一條】虛擬DOM Diff算法

我們都知道React通過(guò)虛擬DOM機(jī)制可以有效解決復(fù)雜的DOM操作帶來(lái)的性能瓶頸,并且每當(dāng)數(shù)據(jù)變化時(shí),React都會(huì)重新構(gòu)建整個(gè)DOM樹,然后將當(dāng)前DOM樹和上次DOM樹進(jìn)行對(duì)比,然后將需要變化的部分進(jìn)行實(shí)際更新。那么,需要變化的部分細(xì)化到什么地步,虛擬DOM是又如何運(yùn)作的?不懂這些對(duì)我們實(shí)際使用react并不會(huì)有什么困擾,但是理解這些對(duì)我們實(shí)現(xiàn)自定義組件時(shí)如何進(jìn)一步優(yōu)化性能具有指導(dǎo)意義。

Diff算法復(fù)雜度為O(n),這基于下面兩個(gè)假設(shè):

  • 兩個(gè)相同組件產(chǎn)生類似的DOM結(jié)構(gòu),不同的組件產(chǎn)生不同的DOM結(jié)構(gòu);
  • 對(duì)于同一層次的一組子節(jié)點(diǎn),他們可以通過(guò)唯一ID進(jìn)行區(qū)分。

逐層進(jìn)行節(jié)點(diǎn)比較

React按樹結(jié)構(gòu)從上到下依次遍歷,逐層比較。它只會(huì)對(duì)相同顏色框內(nèi)的DOM節(jié)點(diǎn)比較,也就是同一個(gè)父節(jié)點(diǎn)下的所有子節(jié)點(diǎn)。

Diff算法逐層進(jìn)行比較

當(dāng)該位置前后節(jié)點(diǎn)類型不同時(shí)

當(dāng)樹中的同一位置前后輸出了不同類型的節(jié)點(diǎn):直接刪除前面的節(jié)點(diǎn)(不管該節(jié)點(diǎn)是否有子節(jié)點(diǎn)),然后創(chuàng)建并插入新的節(jié)點(diǎn)。同樣的,同一位置前后輸出不同組件時(shí)也是直接銷毀第一個(gè)組件,然后創(chuàng)建新組件并插入。



對(duì)于上圖,Diff算法的操作是

A.destroy();
A=new A();
A.append(new B());
A.append(new C());
D.append(new D());

假設(shè)這些節(jié)點(diǎn)都是組件,那么從生命周期來(lái)理解的話就是:

A will unmount.
A is created.
B is created.
C is created.
B did mount.
C did mount.
A did mount.
D is updated.
Root is updated.

由上面簡(jiǎn)單的例子可見(jiàn),Diff算法在同一位置不論是對(duì)組件還是組件內(nèi)的節(jié)點(diǎn)只要類型不同直接刪除加重建,這樣從上到下遍歷一次搞定。那么對(duì)于我們平日的使用:能用display:none之類的class解決的事,就不要去替換修改DOM結(jié)構(gòu),(這點(diǎn)我們?cè)趈query實(shí)踐中也基本都做到了)。另外對(duì)于節(jié)點(diǎn)列表的操作(一般涉及到增、刪、查、改、排序),要給每個(gè)節(jié)點(diǎn)去設(shè)置唯一的標(biāo)識(shí)(key),?這可以幫助React定位到正確的節(jié)點(diǎn)進(jìn)行比較,從而大幅減少DOM操作次數(shù),提高性能。

Flux應(yīng)用程序架構(gòu)

一個(gè)小的組件寫起來(lái)好像并不太不煩,譬如先前提到的一個(gè)實(shí)現(xiàn)評(píng)論功能的組件。但是內(nèi)容提交后如何在上面內(nèi)容區(qū)域顯示呢,也就是組件之間如何通信?

Flux的解決方案是讓數(shù)據(jù)流變成單向,引入Store、Action、Action Creators和Dispatcher等概念來(lái)管理信息流,完全面向View,并且始終是整體刷新的思路。如下圖所示:

單項(xiàng)數(shù)據(jù)流

這顯然是一個(gè)鏈?zhǔn)椒磻?yīng),Action觸發(fā)Store的更新,Store的更新又出發(fā)view的重新渲染。得益于React的View每次更新都是整體刷新的思路,我們可以完全不必關(guān)心Store的變化細(xì)節(jié),只需要監(jiān)聽Store的onChange事件,每次變化都觸發(fā)View的re-render。

回到提交評(píng)論的使用場(chǎng)景(使用Flux):
1、兩個(gè)組件:評(píng)論列表和評(píng)論框
2、所有組件綁定Store(存儲(chǔ)了評(píng)論數(shù)據(jù))
3、Action Creator向服務(wù)器發(fā)送請(qǐng)求
4、Store中監(jiān)聽action,更新自身數(shù)據(jù),然后觸發(fā)view的更新。

流程圖

組件之間需要共享的狀態(tài)放到Store中進(jìn)行維護(hù),Store中的狀態(tài)改變時(shí),就會(huì)發(fā)布o(jì)nChange事件,訂閱了這個(gè)事件的View就會(huì)執(zhí)行重新渲染,通過(guò)這樣一種發(fā)布——訂閱模式就實(shí)現(xiàn)了從Store到View的數(shù)據(jù)綁定。

那么當(dāng)View接收接收用戶交互后(譬如對(duì)評(píng)論點(diǎn)贊),又如何將新狀態(tài)存入Store?

View接收用戶的輸入之后,產(chǎn)生一個(gè)特定的action,然后通過(guò)Dispatcher分發(fā)處理去更新Store,重新渲染view,這樣也就形成了一個(gè)閉環(huán)的單向流動(dòng)。Dispatcher是全局唯一的,也就是所有action的集中處理中心,而每個(gè)action處理函數(shù)都能接收到所有的action,即所有的action能力都相同,只是當(dāng)前在執(zhí)行的任務(wù)可能不同。另外為了使View更加純粹,將Action的創(chuàng)建邏輯也單拿出去放到了Action Creators中。

Flux的標(biāo)準(zhǔn)實(shí)現(xiàn)非常簡(jiǎn)單,因此還衍生出了很多第三方實(shí)現(xiàn),而如今最為火熱的應(yīng)該屬于Redux,它采用了函數(shù)式編程的思想來(lái)維護(hù)整個(gè)應(yīng)用程序的狀態(tài)。

【參考】
http://www.infoq.com/cn/articles/react-flux?utm_campaign=rightbar_v2&utm_source=infoq&utm_medium=articles_link&utm_content=link_text

http://camsong.github.io/redux-in-chinese/index.html

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 230,431評(píng)論 6 544
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 99,637評(píng)論 3 429
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人,你說(shuō)我怎么就攤上這事?!?“怎么了?”我有些...
    開封第一講書人閱讀 178,555評(píng)論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我,道長(zhǎng),這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,900評(píng)論 1 318
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 72,629評(píng)論 6 412
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 55,976評(píng)論 1 328
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,976評(píng)論 3 448
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 43,139評(píng)論 0 290
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 49,686評(píng)論 1 336
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 41,411評(píng)論 3 358
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 43,641評(píng)論 1 374
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 39,129評(píng)論 5 364
  • 正文 年R本政府宣布,位于F島的核電站,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 44,820評(píng)論 3 350
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,233評(píng)論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,567評(píng)論 1 295
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 52,362評(píng)論 3 400
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 48,604評(píng)論 2 380

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

  • 一、CMS管理系統(tǒng)功能 CMS是ContentManagementSystem的縮寫,意為"內(nèi)容管理系統(tǒng)"。 CM...
    默默先生Alec閱讀 5,393評(píng)論 0 7
  • 原教程內(nèi)容詳見(jiàn)精益 React 學(xué)習(xí)指南,這只是我在學(xué)習(xí)過(guò)程中的一些閱讀筆記,個(gè)人覺(jué)得該教程講解深入淺出,比目前大...
    leonaxiong閱讀 2,853評(píng)論 1 18
  • 作者: 1組 鄭欣欣 侯曉彤 景琪 1 概念 開源硬件指與自由及開放原始碼軟件相同方式設(shè)計(jì)的計(jì)算機(jī)和電子硬件...
    0208_鄭欣欣閱讀 1,014評(píng)論 0 1
  • 公司:廣州市佳音生物科技有限公司 產(chǎn)品名稱:果樂(lè)碗樂(lè) 聯(lián)系人:趙唯...
    唯淋閱讀 676評(píng)論 4 1
  • 時(shí)間太長(zhǎng),什么都有可能會(huì)變;一輩子太短,一件事也有可能做不完?;貞浻肋h(yuǎn)站在背后,你不能拋棄,只能擁抱。 工作挺久的...
    趴趴的卷卷兒閱讀 251評(píng)論 0 0