摘要:應(yīng)用主要的的性能瓶頸來(lái)自于一些冗余的程序處理以及組件中的的過(guò)程。為了避免這種情況,在你的應(yīng)用中盡可能多的讓返回。使用工具將幫助你找到應(yīng)用程序中特定的性能問(wèn)題。這個(gè)工具跟用起來(lái)很像,但是它是專門用來(lái)檢測(cè)應(yīng)用性能的。
這段時(shí)間對(duì)自己寫的React應(yīng)用的性能做了一些分析以及優(yōu)化,發(fā)現(xiàn)項(xiàng)目中產(chǎn)生性能問(wèn)題的原因主要來(lái)自兩個(gè)方面:
大量的數(shù)據(jù)渲染使組件進(jìn)行不必要的diff過(guò)程,導(dǎo)致應(yīng)用卡頓;
部分交互操作頻繁的組件中使用了一些不必要的DOM操作,以及在處理比如scroll事件,resize事件等這類容易導(dǎo)致瀏覽器不停重新渲染的操作時(shí),混雜了大量的計(jì)算以及混亂的DOM操作,導(dǎo)致瀏覽器卡頓。
今天主要想討論的是關(guān)于第一點(diǎn)的優(yōu)化,至于第二點(diǎn),這并不是React為我們帶來(lái)的問(wèn)題,而是我們對(duì)瀏覽器的渲染機(jī)制理解和思考還有所欠缺的表現(xiàn),下次我們?cè)偃ド钊氲奶接戇@個(gè)問(wèn)題。
這篇文章是我在探查原因的時(shí)候,在Medium上看到的。原文地址:Performance optimisations for React applications
先聲明,作者的觀點(diǎn)并不是完全可取的,他在文章中的闡述是基于React與Redux的,但事實(shí)上他并沒(méi)有完全使用Redux的connect()函數(shù),這一點(diǎn)Dan也在Tweet上指出了。不過(guò)即使這樣,對(duì)我們單純的使用React來(lái)說(shuō),也是很有意義的。針對(duì)Dan的觀點(diǎn),作者也在文章的評(píng)論中進(jìn)行了補(bǔ)充。
TL;DR;
React應(yīng)用主要的的性能瓶頸來(lái)自于一些冗余的程序處理以及組件中的DOM diff的過(guò)程。為了避免這種情況,在你的應(yīng)用中盡可能多的讓shouldComponentUpdate返回false。
并且你還要加快這個(gè)過(guò)程:
shouldComponentUpdate中的條件判斷應(yīng)該盡可能的快
shouldComponentUpdate中的條件判斷要盡可能的簡(jiǎn)單
以下是正文
是什么造成了React應(yīng)用的性能瓶頸?不需要更新DOM的冗余處理
對(duì)大量不需要更新的DOM節(jié)點(diǎn)進(jìn)行diff計(jì)算(雖然Diff算法是使React應(yīng)用表現(xiàn)良好的關(guān)鍵,但這些計(jì)算并不能夠完全忽略不計(jì))
React中默認(rèn)的渲染方式是什么?讓我來(lái)研究下React是如何渲染一個(gè)組件的
首次render對(duì)于首次渲染來(lái)說(shuō),所有的節(jié)點(diǎn)都應(yīng)當(dāng)被渲染(綠色的表示被渲染的節(jié)點(diǎn))
圖中的每個(gè)節(jié)點(diǎn)都被渲染了,我們的應(yīng)用目前處于初始狀態(tài)。
我們想要更新一段數(shù)據(jù),而跟這個(gè)數(shù)據(jù)相關(guān)的只有一個(gè)節(jié)點(diǎn)。
我們只想渲染到達(dá)葉子節(jié)點(diǎn)的關(guān)鍵路徑。
如果我們什么都不做的話,React默認(rèn)會(huì)這樣做:(orange = waste)
所有的節(jié)點(diǎn)都被渲染了。
每個(gè)React組件中都有一個(gè)shouldComponentUpdate(nextProps, nextState)方法。它返回一個(gè)Bool值,當(dāng)組件應(yīng)當(dāng)被渲染時(shí)返回true,不應(yīng)當(dāng)被渲染時(shí)返回false。當(dāng)return false時(shí),組件中的render方法壓根不會(huì)被執(zhí)行。然而在React中,即便你沒(méi)有明確的定義shouldComponentUpdate方法,shouldComponentUpdate還是會(huì)默認(rèn)返回True。
// default behaviour shouldComponentUpdate(nextProps, nextState) { return true; }
這意味著,當(dāng)我們對(duì)默認(rèn)行為不做任何修改時(shí),每次改動(dòng)頂層的props,整個(gè)應(yīng)用的所有組件都會(huì)執(zhí)行其render方法。這就是產(chǎn)生性能問(wèn)題的原因。
那么我們?nèi)绾螌?shí)現(xiàn)理想化的更新操作呢?在你的應(yīng)用中盡可能多的讓shouldComponentUpdate返回false。
并且你還要加快這個(gè)過(guò)程:
shouldComponentUpdate中的條件判斷應(yīng)該盡可能的快
shouldComponentUpdate中的條件判斷要盡可能的簡(jiǎn)單
加快shouldComponentUpdate中的條件判斷理想化的情況中,我們不應(yīng)該在shouldComponentUpdate函數(shù)中進(jìn)行深度比較,因?yàn)樯疃缺容^是比較消耗資源的一件事,特別是我們的數(shù)據(jù)結(jié)構(gòu)嵌套特別深,數(shù)據(jù)量特別大的時(shí)候。
class Item extends React.Component { shouldComponentUpdate(nextProps) { // expensive! return isDeepEqual(this.props, nextProps); } // ... }
一個(gè)可供替代的方法是在一個(gè)數(shù)據(jù)發(fā)生改變時(shí),修改對(duì)象的引用而不是它的值。
const newValue = { ...oldValue // any modifications you want to do }; // fast check - only need to check references newValue === oldValue; // false // you can also use the Object.assign syntax if you prefer const newValue2 = Object.assign({}, oldValue); newValue2 === oldValue; // false
可以把這個(gè)技巧用在Redux中的reducer中:
// in this Redux reducer we are going to change the description of an item export default (state, action) => { if(action.type === "ITEM_DESCRIPTION_UPDATE") { const {itemId, description} = action; const items = state.items.map(item => { // action is not relevant to this item - we can return the old item unmodified if(item.id !== itemId) { return item; } // we want to change this item // this will keep the "value" of the old item but // return a new object with an updated description return { ...item, description }; }); return { ...state, items }; } return state; }
如果你采用了這種方式,那么在你的shouldComponentUpdate方法中只需要檢查對(duì)象的引用就可以。
// super fast - all you are doing is checking references! shouldComponentUpdate(nextProps) { return !isObjectEqual(this.props, nextProps); }
下面是isObjectEqual函數(shù)的一種簡(jiǎn)易實(shí)現(xiàn):
const isObjectEqual = (obj1, obj2) => { if(!isObject(obj1) || !isObject(obj2)) { return false; } // are the references the same? if (obj1 === obj2) { return true; } // does it contain objects with the same keys? const item1Keys = Object.keys(obj1).sort(); const item2Keys = Object.keys(obj2).sort(); if (!isArrayEqual(item1Keys, item2Keys)) { return false; } // does every object in props have the same reference? return item2Keys.every(key => { const value = obj1[key]; const nextValue = obj2[key]; if (value === nextValue) { return true; } // special case for arrays - check one level deep return Array.isArray(value) && Array.isArray(nextValue) && isArrayEqual(value, nextValue); }); }; const isArrayEqual = (array1 = [], array2 = []) => { if (array1 === array2) { return true; } // check one level deep return array1.length === array2.length && array1.every((item, index) => item === array2[index]); };讓shouldComponentUpdate中的條件判斷更簡(jiǎn)單
下面是一個(gè)比較復(fù)雜的shouldComponentUpdate函數(shù):
// Data structure with good separation of concerns (normalised data) const state = { items: [ { id: 5, description: "some really cool item" } ], // an object to represent the users interaction with the system interaction: { selectedId: 5 } };
這樣的數(shù)據(jù)結(jié)構(gòu)讓你的shouldComponentUpdate函數(shù)變得復(fù)雜:
import React, {Component, PropTypes} from "react"; class List extends Component { propTypes = { items: PropTypes.array.isRequired, interaction: PropTypes.object.isRequired } shouldComponentUpdate (nextProps) { // have any of the items changed? if(!isArrayEqual(this.props.items, nextProps.items)){ return true; } // everything from here is horrible. // if interaction has not changed at all then when can return false (yay!) if(isObjectEqual(this.props.interaction, nextProps.interaction)){ return false; } // at this point we know: // 1. the items have not changed // 2. the interaction has changed // we need to find out if the interaction change was relevant for us const wasItemSelected = this.props.items.any(item => { return item.id === this.props.interaction.selectedId }); const isItemSelected = nextProps.items.any(item => { return item.id === nextProps.interaction.selectedId }); // return true when something has changed // return false when nothing has changed return wasItemSelected !== isItemSelected; } render() {問(wèn)題1:龐大的shouldComponentUpdate函數(shù){this.props.items.map(item => { const isSelected = this.props.interaction.selectedId === item.id; return (} }- ); })}
從上面的例子就可以看出來(lái),即便是那么一小段很簡(jiǎn)單的數(shù)據(jù)結(jié)構(gòu),shouldConponentUpdate函數(shù)依然有如此繁雜的處理。這是因?yàn)檫@個(gè)函數(shù)需要了解數(shù)據(jù)結(jié)構(gòu),以及每個(gè)數(shù)據(jù)之間又怎樣的關(guān)聯(lián)。所以說(shuō),shouldComponentUpdate函數(shù)的大小和復(fù)雜度,是由數(shù)據(jù)結(jié)構(gòu)決定的。
這很容易引起兩個(gè)錯(cuò)誤:
不該返回false時(shí),返回了false(狀態(tài)在程序中沒(méi)有被正確處理,導(dǎo)致視圖不更新)
不該返回true時(shí),返回了true(視圖每次都更新,引起了性能問(wèn)題)
何必為難自己呢?你想要讓你的程序足夠簡(jiǎn)單,而不需要仔細(xì)考慮這些數(shù)據(jù)之間的關(guān)系。(所以,想要讓程序變得簡(jiǎn)單,一定要設(shè)計(jì)好數(shù)據(jù)結(jié)構(gòu))
問(wèn)題2:高度耦合的父子組件應(yīng)用普遍都是耦合度越低越好(組件之間要盡可能的不互相依賴)。父組件不應(yīng)該試圖去理解子組件是如何運(yùn)行的。這允許你修改子組件的行為,而父組件不需要知道更改(假定子組件的PropTypes不變)。這同樣意味著子組件可以獨(dú)立運(yùn)行,而不需要父組件嚴(yán)格的控制它的行為。
規(guī)范化你的數(shù)據(jù)結(jié)構(gòu)通過(guò)規(guī)范化你的數(shù)據(jù)結(jié)構(gòu),你可以很方便的只通過(guò)判斷引用是否更改來(lái)判斷視圖是否需要更新。
const state = { items: [ { id: 5, description: "some really cool item", // interaction now lives on the item itself interaction: { isSelected: true } } ] };
這樣的數(shù)據(jù)結(jié)構(gòu)讓你在shouldComponentUpdate函數(shù)中的更新檢測(cè)更加簡(jiǎn)單。
import React, {Component, PropTypes} from "react"; class List extends Component { propTypes = { items: PropTypes.array.isRequired } shouldComponentUpdate (nextProps) { // so easy return isObjectEqual(this.props, nextProps); } render() {{this.props.items.map(item => { return (} }- ); })}
如果你想要更新其中的一個(gè)數(shù)據(jù),比如interaction,你只需要更新整個(gè)對(duì)象的引用就可以了。
// redux reducer export default (state, action) => { if(action.type === "ITEM_SELECT") { const {itemId} = action; const items = state.items.map(item => { if(item.id !== itemId) { return item; } // changing the reference to the whole object return { ...item, interaction: { isSelected: true } } }); return { ...state, items }; } return state; };引用檢查和動(dòng)態(tài)props
先看一個(gè)創(chuàng)建動(dòng)態(tài)props的例子
class Foo extends React.Component { render() { const {items} = this.props; // this object will have a new reference every time const newData = { hello: "world" }; return- } } class Item extends React.Component { // this will always return true as `data` will have a new reference every time // even if the objects have the same value shouldComponentUpdate(nextProps) { return isObjectEqual(this.props, nextProps); } }
通常我們不在組件中創(chuàng)建新的props,只是將它傳遞下去。
然而下面這種內(nèi)部循環(huán)的方式卻越來(lái)越普遍了:
class List extends React.Component { render() { const {items} = this.props;{items.map((item, index) => { // this object will have a new reference every time const newData = { hello: "world", isFirst: index === 0 }; return} }- })}
這是在創(chuàng)建函數(shù)中經(jīng)常使用的。
import myActionCreator from "./my-action-creator"; class List extends React.Component { render() { const {items, dispatch} = this.props;解決這個(gè)問(wèn)題的策略{items.map(item => { // this function will have a new reference every time const callback = () => { dispatch(myActionCreator(item)); } return} }- })}
避免在組件內(nèi)部創(chuàng)建動(dòng)態(tài)props(改善你的數(shù)據(jù)結(jié)構(gòu),使props可以被直接用來(lái)傳遞)
將動(dòng)態(tài)props當(dāng)做滿足===不等式的類型傳遞(eg: Bool, Number, String)
const bool1 = true; const bool2 = true; bool1 === bool2; // true const string1 = "hello"; const string2 = "hello"; string1 === string2; // true
如果你真的需要傳遞一個(gè)動(dòng)態(tài)對(duì)象,你可以傳遞一個(gè)對(duì)象的字符串表示,并且這個(gè)字符串應(yīng)當(dāng)可以在子組件中重新解讀為相應(yīng)的對(duì)象。
render() { const {items} = this.props;特例:函數(shù){items.map(item => { // will have a new reference every time const bad = { id: item.id, type: item.type }; // equal values will satify strict equality "===" const good = `${item.id}::${item.type}`; return}- })}
盡量不要傳遞函數(shù)。在子組件需要時(shí)才去觸發(fā)相應(yīng)的actions。這樣做還有一個(gè)好處是將業(yè)務(wù)邏輯與組件分離開(kāi)來(lái)。
忽略shouldComponentUpdate函數(shù)中對(duì)functions的檢查,因?yàn)槲覀儫o(wú)法知曉函數(shù)的值是否發(fā)生改變。
創(chuàng)建一個(gè)不可變數(shù)據(jù)與函數(shù)的映射。你可以在執(zhí)行componentWillReveiveProps函數(shù)時(shí),把這個(gè)映射放到state中。這樣的話每次render時(shí)將不會(huì)得到一個(gè)新的引用,便于執(zhí)行在shouldComponentUpdate時(shí)的引用檢查。這個(gè)方法比較麻煩,因?yàn)樾枰S護(hù)和更新函數(shù)列表。
創(chuàng)建一個(gè)有正確this綁定的中間組件。這樣也并不理想,因?yàn)樵诮M件的層次結(jié)構(gòu)中引入了冗余層。(實(shí)際上作者的意思是將函數(shù)的定義從render函數(shù)中移出,這樣每次的render就不會(huì)創(chuàng)建新的引用了)
避免每次執(zhí)行render函數(shù)時(shí),都創(chuàng)建一個(gè)新的函數(shù)。
關(guān)于第四點(diǎn)的例子:
// introduce another layer "ListItem"工具class ListItem extends React.Component { // this will always have the correct this binding as it is tied to the instance // thanks es7! const callback = () => { dispatch(doSomething(item)); } render() { return
// you can create the correct this bindings in here - } }
上面列出的所有規(guī)則和技術(shù)都是通過(guò)使用性能測(cè)量工具發(fā)現(xiàn)的。 使用工具將幫助你找到應(yīng)用程序中特定的性能問(wèn)題。
console.time這個(gè)工具相當(dāng)簡(jiǎn)單。
開(kāi)始計(jì)時(shí)
程序運(yùn)行
結(jié)束計(jì)時(shí)
一個(gè)很棒的方式是用Redux的中間件來(lái)測(cè)試性能。
export default store => next => action => { console.time(action.type); // `next` is a function that takes an "action" and sends it through to the "reducers" // this will result in a re-render of your application const result = next(action); // how long did the render take? console.timeEnd(action.type); return result; };
用這個(gè)方法,你可以記錄每個(gè)操作及其在應(yīng)用程序中渲染所花費(fèi)的時(shí)間。 你可以快速查看是哪些操作需要耗費(fèi)很多時(shí)間來(lái)執(zhí)行,這給我們提供了解決性能問(wèn)題的一個(gè)起點(diǎn)。 有了這個(gè)時(shí)間值,還有助于我們查看我們對(duì)代碼的更改對(duì)應(yīng)用程序產(chǎn)生的影響。
React.perf這個(gè)工具跟console.time用起來(lái)很像,但是它是專門用來(lái)檢測(cè)React應(yīng)用性能的。
Perf.start
程序運(yùn)行
Perf.stop
依然是用Redux的中間件舉個(gè)例子
import Perf from "react-addons-perf"; export default store => next => action => { const key = `performance:${action.type}`; Perf.start(); // will re-render the application with new state const result = next(action); Perf.stop(); console.group(key); console.info("wasted"); Perf.printWasted(); // any other Perf measurements you are interested in console.groupEnd(key); return result; };
與console.time方法類似,您可以查看每個(gè)操作的表現(xiàn)數(shù)據(jù)。 有關(guān)React性能插件的更多信息,請(qǐng)參閱此處
瀏覽器開(kāi)發(fā)者工具CPU分析器的Flame圖也有助于在應(yīng)用程序中查找性能問(wèn)題。
Flame圖顯示性能展示文件中每毫秒代碼的JavaScript堆棧的狀態(tài)。 這給你一個(gè)方法來(lái)確切地知道哪個(gè)函數(shù)在記錄期間的哪個(gè)點(diǎn)執(zhí)行,運(yùn)行了多長(zhǎng)時(shí)間,以及是從哪里被調(diào)用的 - Mozilla
Firefox: see here
Chrome: see here
感謝閱讀以及一切能讓React應(yīng)用性能提高的方式!
作者的補(bǔ)充:在檢查每個(gè)子組件的列表組件上使用shouldComponentUpdate(),并不是非常有用。
當(dāng)你有很多大列表的時(shí)候,這個(gè)方法是很有用的。能夠完全跳過(guò)列表的重新渲染時(shí)一個(gè)巨大的勝利。但是如果你的應(yīng)用中只有一個(gè)大列表,那么這樣做其實(shí)沒(méi)有任何效果,因?yàn)槟愕娜魏尾僮鞫际腔谶@個(gè)列表的,意味著列表中的數(shù)據(jù)肯定會(huì)有所改變,那么你完全可以跳過(guò)對(duì)更新條件的檢查。
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/81538.html
摘要:譯文地址譯唯快不破應(yīng)用的個(gè)優(yōu)化步驟前端的逆襲知乎專欄原文地址時(shí)過(guò)境遷,應(yīng)用比以往任何時(shí)候都更具交互性。使用負(fù)載均衡方案我們?cè)谥坝懻摼彺娴臅r(shí)候簡(jiǎn)要提到了內(nèi)容分發(fā)網(wǎng)絡(luò)。換句話說(shuō),元素的串形訪問(wèn)會(huì)削弱負(fù)載均衡器以最佳形式 歡迎關(guān)注知乎專欄 —— 前端的逆襲歡迎關(guān)注我的博客,知乎,GitHub。 譯文地址:【譯】唯快不破:Web 應(yīng)用的 13 個(gè)優(yōu)化步驟 - 前端的逆襲 - 知乎專欄原文地...
摘要:雖然有著各種各樣的不同,但是相同的是,他們前端優(yōu)化不完全指南前端掘金篇幅可能有點(diǎn)長(zhǎng),我想先聊一聊閱讀的方式,我希望你閱讀的時(shí)候,能夠把我當(dāng)作你的競(jìng)爭(zhēng)對(duì)手,你的夢(mèng)想是超越我。 如何提升頁(yè)面渲染效率 - 前端 - 掘金Web頁(yè)面的性能 我們每天都會(huì)瀏覽很多的Web頁(yè)面,使用很多基于Web的應(yīng)用。這些站點(diǎn)看起來(lái)既不一樣,用途也都各有不同,有在線視頻,Social Media,新聞,郵件客戶端...
摘要:我的教程可能也會(huì)幫你一把其他的二分法展示型組件和容器型組件這種分類并非十分嚴(yán)格,這是按照它們的目的進(jìn)行分類。在我看來(lái),展示型組件往往是無(wú)狀態(tài)的純函數(shù)組件,容器型組件往往是有狀態(tài)的純類組件。不要把展示型組件和容器型組件的劃分當(dāng)成教條。 本文譯自Presentational and Container Components,文章的作者是Dan Abramov,他同時(shí)也是Redux和Crea...
摘要:自己英語(yǔ)一般,水平有限,獻(xiàn)上原文地址,還有我翻譯的中文地址,歡迎大家勘誤下面是自己的一點(diǎn)感想先說(shuō)一下,我們知道,前端優(yōu)化有這么幾步,第一步首先呢我們知道,一個(gè)應(yīng)用要依賴好多條文件,而瀏覽器加載完一條,要執(zhí)行完這條才加載下一條,所以呢,就很慢 自己英語(yǔ)一般,水平有限,獻(xiàn)上原文地址,還有我翻譯的中文地址,歡迎大家勘誤 下面是自己的一點(diǎn)感想 先說(shuō)一下webpack,我們知道,前端優(yōu)化有這么幾...
摘要:對(duì)此沒(méi)有任何限制,它不關(guān)心這個(gè)。一種控制變化的辦法是不可改變的,持久化的數(shù)據(jù)結(jié)構(gòu)。總結(jié)檢測(cè)變化時(shí)開(kāi)發(fā)中的核心問(wèn)題,而框架們以各種方式解決這個(gè)問(wèn)題。因?yàn)榻M件內(nèi)的變化是不被允許的。 AngularJS:臟檢查 我不知道什么更新了,所以當(dāng)更新的時(shí)候,我只能檢查所有的東西。 AngularJS 類似于 Ember,當(dāng)狀態(tài)改變的時(shí)候,必須人工去處理。但不同的是,AngularJS 從不同的角度來(lái)...
閱讀 1658·2019-08-30 15:55
閱讀 981·2019-08-30 15:44
閱讀 873·2019-08-30 10:48
閱讀 2046·2019-08-29 13:42
閱讀 3190·2019-08-29 11:16
閱讀 1269·2019-08-29 11:09
閱讀 2060·2019-08-26 11:46
閱讀 620·2019-08-26 11:44