摘要:本文主要介紹關(guān)鍵渲染路徑與網(wǎng)絡(luò)兩個(gè)方面的性能優(yōu)化并提供,篇幅較長(zhǎng)建議電腦觀看。百度統(tǒng)計(jì)代碼注意,的腳本不會(huì)被阻塞,完成后立即執(zhí)行,但是有可能會(huì)阻塞關(guān)鍵渲染路徑。
本文主要介紹“關(guān)鍵渲染路徑”與“網(wǎng)絡(luò)”兩個(gè)方面的性能優(yōu)化并提供demo,篇幅較長(zhǎng)建議電腦觀看。
前端優(yōu)化的方面太多,本文介紹的僅僅是其中的一部分,力求涵蓋“關(guān)鍵渲染路徑”的方方面面,及一些不常被提到的“網(wǎng)絡(luò)優(yōu)化”部分。
渲染頁(yè)面過(guò)程測(cè)試環(huán)境如無(wú)特殊說(shuō)明均為Chrome 57
瀏覽器從打開(kāi)一個(gè)URL到渲染完頁(yè)面共有:
下載HTML文檔
下載HTML文檔中的css
下載Js文件
執(zhí)行js腳本
下載其他資源
通過(guò)HTML文檔構(gòu)建DOM(Parse HTML)
通過(guò)CSS文件構(gòu)建CSSOM(Parse CSS)
通過(guò)DOM與CSSOM計(jì)算render tree
根據(jù)render tree進(jìn)行繪制,計(jì)算各個(gè)元素位置與大小(Layout)
對(duì)頁(yè)面進(jìn)行上色,渲染為最終顯示的像素(Paint)
第一次完成Paint稱為“初次渲染”,這時(shí)候用戶就能看到render tree里面的東西了。而完成初次渲染的過(guò)程稱為“關(guān)鍵渲染路徑”,關(guān)鍵渲染路徑上需要加載的資源叫做“關(guān)鍵資源”
這個(gè)過(guò)程很多很復(fù)雜,其中的依賴關(guān)系也很復(fù)雜,筆者嘗試畫圖來(lái)表示,但是實(shí)在是沒(méi)畫出來(lái),所以還是用文字來(lái)表述吧:
引入的資源,哪怕被阻塞(比如被js腳本阻塞后續(xù)link標(biāo)簽),瀏覽器依舊會(huì)智能的預(yù)先加載它們(但是不執(zhí)行)
“CSS文件的加載”會(huì)阻塞“Js文件執(zhí)行”。若CSS引用在Js文件之前,“加載CSS文件”會(huì)阻塞“Js文件執(zhí)行”。即CSS文件未加載解析完成前,js文件不會(huì)得到執(zhí)行。因?yàn)閖s有可能會(huì)修改CSSOM。帶有async和defer屬性的script不受限制。
Parse HTML的解析是增量的,因此瀏覽器可以邊下載HTML邊構(gòu)建DOM樹(shù)
“CSS文件的加載”會(huì)阻塞“Layout”。若頁(yè)面有正在加載的CSS文件,在CSS文件加載完之前,瀏覽器不會(huì)對(duì)頁(yè)面進(jìn)行Layout,這是為了防止樣式突變帶來(lái)的抖動(dòng)
“加載Js文件”會(huì)阻塞“Parse HTML”,這個(gè)估計(jì)大家都知道了,因?yàn)閖s可以通過(guò)document.write修改HTML文檔流
“Js文件執(zhí)行”會(huì)幾乎會(huì)阻塞所有東西,包括Layout
比較有意思的是,字體的加載會(huì)阻塞局部的渲染。若某一段文本的字體使用了一個(gè)尚未加載完的字體,這段文本則先不會(huì)被Paint,直到字體加載完或者超過(guò)某個(gè)時(shí)間(通常是3秒)文本才會(huì)突然顯示。
瀏覽器為了避免FOUT(Flash Of Unstyled Text),會(huì)盡量等待字體加載完成后,再顯示應(yīng)用了該字體的內(nèi)容。只有當(dāng)字體超過(guò)一段時(shí)間仍未加載成功時(shí),瀏覽器才會(huì)降級(jí)使用系統(tǒng)字體。每個(gè)瀏覽器都規(guī)定了自己的超時(shí)時(shí)間(Chrome是3秒)。但這也帶來(lái)了FOIT(Flash Of Invisible Text)問(wèn)題。內(nèi)容無(wú)法盡快地被展示,導(dǎo)致空白
一些Demo來(lái)解釋瀏覽器渲染流程CSS會(huì)阻塞Layout:Demo
Title Hello
World
CSS會(huì)阻塞Js執(zhí)行:Demo
Title Hello
World
Js執(zhí)行會(huì)阻塞關(guān)鍵渲染路徑,哪怕是defer還是async:Demo
Title Hello
World
Foot會(huì)阻塞局部渲染,但是智能的瀏覽器會(huì)給他設(shè)定一個(gè)上限,一般是3秒鐘:Demo
CSS篇優(yōu)化策略Title Hello
World
優(yōu)化核心概念是:將初次渲染不需要的CSS想辦法剝離出關(guān)鍵渲染路徑
如果僅僅是為了提前初次渲染時(shí)間而進(jìn)行優(yōu)化,將頁(yè)面必備的CSS剝離關(guān)鍵渲染路徑而造成樣式突變導(dǎo)致頁(yè)面抖動(dòng),則得不償失了
使用link/style的media屬性對(duì)某些媒體查詢條件觸發(fā)后才使用的css,可以在link標(biāo)簽中加入media屬性,如下:
此樣式表仍會(huì)加載。當(dāng)瀏覽器環(huán)境不匹配媒體查詢條件時(shí),該樣式表不會(huì)阻塞渲染。我們可針對(duì)不同媒體環(huán)境拆分CSS文件,并為link標(biāo)簽添加媒體查詢,避免為了加載非關(guān)鍵CSS資源,而阻塞初次渲染
使用DOM API添加CSS可以使用js代碼來(lái)添加css
var style = document.createElement("link"); style.rel = "stylesheet"; style.href = "index.css"; document.head.appendChild(style);使用resoure hint規(guī)范的preload
將link標(biāo)簽的rel屬性設(shè)置為preload,瀏覽器遇到遇到標(biāo)記為preload的link時(shí),會(huì)開(kāi)始加載它,但是由于rel不是stylesheet,因此不會(huì)阻塞渲染。
然后在適當(dāng)?shù)臅r(shí)候,在rel改為stylesheet,即可應(yīng)用此樣式。
但是這個(gè)屬性兼容性比較差,詳細(xì)可以參考這里。不過(guò)有一個(gè)polyfill可以用loadCSS,原理是通過(guò)DOM API插入樣式資源。
這個(gè)屬性的使用情景有些偏,也可能是我理解問(wèn)題:
當(dāng)使用preload引入css文件時(shí),實(shí)際上證明這個(gè)頁(yè)面根本不需要這個(gè)css,它有可能是打印樣式,或者是響應(yīng)式網(wǎng)站的另一套css代碼。但是,使用preload屬性,瀏覽器反而會(huì)預(yù)先加載它,也就是說(shuō),在window.onload之前,用戶將耗費(fèi)了網(wǎng)絡(luò)資源在加載一個(gè)暫時(shí)不需要的樣式。網(wǎng)絡(luò)資源不可能是無(wú)限的,也就是說(shuō)這個(gè)css會(huì)占用頁(yè)面其他資源比如圖片的網(wǎng)絡(luò)資源。
詢問(wèn)瓜瓜老師本人后,瓜瓜老師說(shuō):
舉個(gè)例子。第三屏有個(gè)廣告版,它的樣式
這樣確實(shí)這個(gè)css的緊急程度就介于關(guān)鍵渲染路徑的css與頁(yè)面圖片之間了,不過(guò)貌似這個(gè)情景很受限。
JS篇優(yōu)化策略 使用defer延遲腳本執(zhí)行當(dāng)script標(biāo)簽擁有defer屬性時(shí),該腳本會(huì)被推遲到整個(gè)HTML文檔解析完后,再開(kāi)始執(zhí)行。因此將腳本放在head中,可以提早瀏覽器對(duì)腳本文件的加載,但是卻不會(huì)阻塞parse HTML。
注意,defer的腳本不會(huì)被css阻塞,parse HTML完成后立即執(zhí)行,但是有可能會(huì)阻塞關(guān)鍵渲染路徑。為什么說(shuō)有可能呢,假如腳本文件在render tree生成前加載完畢,則會(huì)開(kāi)始執(zhí)行,執(zhí)行過(guò)程中會(huì)阻塞關(guān)鍵渲染路徑。請(qǐng)參考這個(gè)Demo
被defer的腳本,在執(zhí)行時(shí)會(huì)嚴(yán)格按照在HTML文檔中出現(xiàn)的順序執(zhí)行,但是實(shí)際上貌似不是這樣,js文件前后文件若有依賴需慎重使用。
使用async延遲腳本執(zhí)行和defer類似,只是當(dāng)js加載完后馬上執(zhí)行,而不在乎parse HTML是否完成,因此假如腳本比css先加載完,也會(huì)阻塞關(guān)鍵渲染路徑。
使用DOM API
據(jù)筆者所知,這是唯一一種100%不會(huì)阻塞關(guān)鍵渲染路徑的js腳本加載方式。通過(guò)DOM API引入的js腳本會(huì)等到頁(yè)面Layout和Paint后再開(kāi)始執(zhí)行,不論你將載入js文件的代碼放在head中還是body后面亦是如此。
其他的優(yōu)化 使用Web Font Loader加載字體若不想讓字體阻塞局部渲染,可使用Web Font Loader
網(wǎng)絡(luò)優(yōu)化篇網(wǎng)絡(luò)優(yōu)化和CSS優(yōu)化策略相同,盡可能讓關(guān)鍵資源提前加載完,所以優(yōu)化時(shí)盡量將以下指標(biāo)壓縮到最低:
關(guān)鍵資源數(shù)
關(guān)鍵資源體積
關(guān)鍵資源網(wǎng)絡(luò)來(lái)回?cái)?shù)
當(dāng)然,如果你的項(xiàng)目使用了先進(jìn)的SPDY或HTTP/2,下面的方法可能并不適用。
優(yōu)化關(guān)鍵資源數(shù)RFC2616規(guī)定同域名同時(shí)只能有 2 個(gè)連接(RFC7230 中無(wú)限制),而???????現(xiàn)代瀏覽器一般允許同域6個(gè)并發(fā)連接。因此,當(dāng)頁(yè)面中有許多需要外鏈的資源(script、link等),瀏覽器最多在每個(gè)域同時(shí)并發(fā)下載6個(gè)。
每一個(gè)請(qǐng)求,若使用域名,則需要額外增加一次DNS查詢時(shí)間(若緩存未過(guò)期會(huì)命中緩存),因此一個(gè)網(wǎng)站過(guò)多的使用不同域名的資源會(huì)額外增加DNS查詢開(kāi)銷,這點(diǎn)在移動(dòng)端非常明顯。
當(dāng)然,每個(gè)請(qǐng)求建立根據(jù)TCP協(xié)議規(guī)定,還需要先進(jìn)行3次捂手才可以建立鏈接。
合并請(qǐng)求盡可能的合并請(qǐng)求,減少網(wǎng)絡(luò)請(qǐng)求數(shù)。這一點(diǎn)可能在其他性能優(yōu)化文章都說(shuō)爛了:
小圖片轉(zhuǎn)base64
合并打包CSS、JS文件
現(xiàn)在的比較流行的webpack就非常擅長(zhǎng)做這種事情
適度使用內(nèi)聯(lián)CSS和Js使用內(nèi)聯(lián)的CSS和JS固然可以減少請(qǐng)求,但是使用內(nèi)聯(lián)也意味著你的CSS和JS將不會(huì)再被瀏覽器緩存,因此要適度的使用內(nèi)聯(lián),內(nèi)聯(lián)不是萬(wàn)能的。
從HTTP協(xié)議下手最佳方案肯定是過(guò)渡到HTTP/2無(wú)疑,但是現(xiàn)在HTTP/2的支持并不算太好,而且各大瀏覽器僅支持TLS下實(shí)現(xiàn)的HTTP/2(說(shuō)白了就是HTTPS),使得HTTP/2的使用存在許些限制。
如果沒(méi)有HTTP/2,或許可以:
使用Keep-Alive可以規(guī)避TCP三次握手的時(shí)間
使用Transfer-Encoding:chunked分塊輸出文件,還記得parse HTML的過(guò)程是增量的嗎?若瀏覽器可以邊下載HTML文件邊解析,豈不美哉?
減少重定向,這個(gè)看上去理所當(dāng)然但是實(shí)際上卻很容易被忽略
適度使用域名散列瀏覽器同域并行下載數(shù)量有限,所以只要多建立幾個(gè)二級(jí)域名就好了,然后合理的分配各個(gè)資源就好了。
假如由于某些不可抗拒原因,關(guān)鍵資源數(shù)是12個(gè),那么只要建立2個(gè)二級(jí)域名分別分配給其中的12個(gè)資源,瀏覽器會(huì)同時(shí)并行下載它們了。
不過(guò),使用域名散列要適度,每一個(gè)域名都需要額外的增加一次DNS查詢時(shí)間。當(dāng)然,DNS本身也有緩存,或許適當(dāng)?shù)脑黾覦NS TTL時(shí)間也是個(gè)不錯(cuò)的主意。
壓縮關(guān)鍵資源體積對(duì)于js、css文件,現(xiàn)在網(wǎng)上現(xiàn)成的壓縮工具一堆,而且應(yīng)用十分廣泛,相信大家都知道了,這里就不多說(shuō)了。
說(shuō)到壓縮,服務(wù)器開(kāi)啟一定的壓縮策略(如gzip)是個(gè)不錯(cuò)的主意,效果拔群,資源大概會(huì)壓縮到原有的1/3左右。
圖片壓縮,這個(gè)需要知道什么情境下適合什么類型的圖片,GIF、JPG、PNG使用情景各不相同,具體可以參考這篇文章:圖片格式那么多,哪種更適合你?
關(guān)鍵資源網(wǎng)絡(luò)來(lái)回?cái)?shù)假如一個(gè)頁(yè)面需要引入2個(gè)CSS才能工作,下面有2種方式
2個(gè)均用link引入
1個(gè)用link引入,在css中import另一個(gè)css
毫無(wú)疑問(wèn)肯定是前者快,因?yàn)榍罢叩木W(wǎng)絡(luò)來(lái)回?cái)?shù)是1,而后者是2。
因此,盡可能將資源加載扁平化,減少關(guān)鍵資源網(wǎng)絡(luò)來(lái)回?cái)?shù)是個(gè)不錯(cuò)的主意。
當(dāng)然,優(yōu)化時(shí)要注意的點(diǎn)也有不少,比如前面提到的瀏覽器同域并發(fā)限制等,需要權(quán)衡使其不要影響到其他的導(dǎo)致初次渲染時(shí)間延后。
一些無(wú)效的優(yōu)化策略使用document.write打印link標(biāo)簽引入css仍會(huì)阻塞初次渲染。
引用奇舞團(tuán)@瓜瓜老師:
PPT:https://ppt.baomitu.com/d/258...
個(gè)人主頁(yè):http://melonh.com/
奇舞團(tuán)@屈屈老師:
PPT:https://ppt.baomitu.com/d/a8a...
個(gè)人主頁(yè):https://imququ.com/
W3C規(guī)范:
https://www.w3.org/TR/2016/RE...
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://specialneedsforspecialkids.com/yun/82726.html
摘要:前端性能優(yōu)化的涉及點(diǎn)從服務(wù)器到協(xié)議再到宿主環(huán)境本身都要有比較深刻的認(rèn)識(shí),業(yè)界目前主要還是以雅虎總結(jié)出來(lái)?xiàng)l前端性能優(yōu)化的黃金軍規(guī)為參考。 歡迎大家前往騰訊云技術(shù)社區(qū),獲取更多騰訊海量技術(shù)實(shí)踐干貨哦~ 導(dǎo)語(yǔ) : 從事前端有6年+的時(shí)間了,從最開(kāi)始的美工到重構(gòu)再到偏向js邏輯開(kāi)發(fā)的前端開(kāi)發(fā),一直在前端這個(gè)行業(yè)里面摸索和學(xué)習(xí),我現(xiàn)在將自己這些年的一個(gè)心得體會(huì)來(lái)個(gè)系統(tǒng)性的梳理寫成一篇關(guān)于性能優(yōu)化...
摘要:前言對(duì)于前端的性能話題,從來(lái)都沒(méi)有斷絕過(guò)。作為一個(gè)前端開(kāi)發(fā)者,性能是我們關(guān)注的指標(biāo)。前端發(fā)展以來(lái),優(yōu)化方式,琳瑯滿目,有雅虎軍規(guī)等。所以,接下來(lái)我會(huì)從三個(gè)方面就前端性能進(jìn)行總結(jié)網(wǎng)絡(luò)方面操作及渲染方面數(shù)據(jù)方面。 前言 對(duì)于前端的性能話題,從來(lái)都沒(méi)有斷絕過(guò)。因?yàn)檫@個(gè)東西沒(méi)有最好,只有更好。而且往往也是業(yè)務(wù)的繁雜程度去決定優(yōu)化程度的。作為一個(gè)前端開(kāi)發(fā)者,性能是我們關(guān)注的指標(biāo)。它直接影響著我們...
摘要:前言對(duì)于前端的性能話題,從來(lái)都沒(méi)有斷絕過(guò)。作為一個(gè)前端開(kāi)發(fā)者,性能是我們關(guān)注的指標(biāo)。前端發(fā)展以來(lái),優(yōu)化方式,琳瑯滿目,有雅虎軍規(guī)等。所以,接下來(lái)我會(huì)從三個(gè)方面就前端性能進(jìn)行總結(jié)網(wǎng)絡(luò)方面操作及渲染方面數(shù)據(jù)方面。 前言 對(duì)于前端的性能話題,從來(lái)都沒(méi)有斷絕過(guò)。因?yàn)檫@個(gè)東西沒(méi)有最好,只有更好。而且往往也是業(yè)務(wù)的繁雜程度去決定優(yōu)化程度的。作為一個(gè)前端開(kāi)發(fā)者,性能是我們關(guān)注的指標(biāo)。它直接影響著我們...
摘要:前言對(duì)于前端的性能話題,從來(lái)都沒(méi)有斷絕過(guò)。作為一個(gè)前端開(kāi)發(fā)者,性能是我們關(guān)注的指標(biāo)。前端發(fā)展以來(lái),優(yōu)化方式,琳瑯滿目,有雅虎軍規(guī)等。所以,接下來(lái)我會(huì)從三個(gè)方面就前端性能進(jìn)行總結(jié)網(wǎng)絡(luò)方面操作及渲染方面數(shù)據(jù)方面。 前言 對(duì)于前端的性能話題,從來(lái)都沒(méi)有斷絕過(guò)。因?yàn)檫@個(gè)東西沒(méi)有最好,只有更好。而且往往也是業(yè)務(wù)的繁雜程度去決定優(yōu)化程度的。作為一個(gè)前端開(kāi)發(fā)者,性能是我們關(guān)注的指標(biāo)。它直接影響著我們...
摘要:端優(yōu)談?wù)勱P(guān)于前端的緩存的問(wèn)題我們都知道對(duì)頁(yè)面進(jìn)行緩存能夠有利于減少請(qǐng)求發(fā)送,從而達(dá)到對(duì)頁(yè)面的優(yōu)化。而作為一名有追求的前端,勢(shì)必要力所能及地優(yōu)化我們前端頁(yè)面的性能。這種方式主要解決了淺談前端中的過(guò)早優(yōu)化問(wèn)題過(guò)早優(yōu)化是萬(wàn)惡之源。 優(yōu)化向:?jiǎn)雾?yè)應(yīng)用多路由預(yù)渲染指南 Ajax 技術(shù)的出現(xiàn),讓我們的 Web 應(yīng)用能夠在不刷新的狀態(tài)下顯示不同頁(yè)面的內(nèi)容,這就是單頁(yè)應(yīng)用。在一個(gè)單頁(yè)應(yīng)用中,往往只有一...
摘要:是一個(gè)極度純凈的上傳插件,通過(guò)簡(jiǎn)單調(diào)整就可以融入到任何項(xiàng)目,支持多文件上傳上傳速率動(dòng)態(tài)控制真實(shí)進(jìn)度監(jiān)控分塊生成分塊上傳校驗(yàn)秒傳暫停取消等。 跨域?qū)W習(xí)筆記 前言: 當(dāng)一個(gè)資源,向與之所在服務(wù)器不同的域或端口請(qǐng)求另一個(gè)資源時(shí),這個(gè)HTTP請(qǐng)求,我們認(rèn)為是跨域的請(qǐng)求。出于安全考慮,瀏覽器會(huì)限制腳本發(fā)起的跨域HTTP請(qǐng)求。 那天后端讓我把token放到http請(qǐng)求頭字段里,說(shuō)是為了和RN端統(tǒng)一...
閱讀 3295·2023-04-26 02:40
閱讀 4648·2021-09-22 15:22
閱讀 1586·2021-09-22 10:02
閱讀 3481·2021-08-11 10:23
閱讀 1392·2019-08-30 15:55
閱讀 2493·2019-08-30 12:48
閱讀 586·2019-08-30 11:04
閱讀 702·2019-08-29 16:29