偷偷摘套内射激情视频,久久精品99国产国产精,中文字幕无线乱码人妻,中文在线中文a,性爽19p

瀏覽器的渲染原理簡介

開發(fā) 前端
看到這個標(biāo)題大家一定會想到這篇神文《How Browsers Work》,這篇文章把瀏覽器的很多細(xì)節(jié)講得很細(xì),而且也被翻譯成了中文。

看到這個標(biāo)題大家一定會想到這篇神文《How Browsers Work》,這篇文章把瀏覽器的很多細(xì)節(jié)講得很細(xì),而且也被翻譯成了中文。為什么我還想寫一篇呢?因為兩個原因,

  1. 這篇文章太長了,閱讀成本太大,不能一口氣讀完。
  2. 花了大力氣讀了這篇文章后可以了解很多,但似乎對工作沒什么幫助。

所以,我準(zhǔn)備寫下這篇文章來解決上述兩個問題。希望你能在上班途中,或是坐馬桶時就能讀完,并能從中學(xué)會一些能用在工作上的東西。

瀏覽器工作大流程

廢話少說,先來看個圖:

從上面這個圖中,我們可以看到那么幾個事:

1)瀏覽器會解析三個東西:

  • 一個是HTML/SVG/XHTML,事實上,Webkit有三個C++的類對應(yīng)這三類文檔。解析這三種文件會產(chǎn)生一個DOM Tree。
  • CSS,解析CSS會產(chǎn)生CSS規(guī)則樹。
  • Javascript,腳本,主要是通過DOM API和CSSOM API來操作DOM Tree和CSS Rule Tree.

2)解析完成后,瀏覽器引擎會通過DOM Tree 和 CSS Rule Tree 來構(gòu)造 Rendering Tree。注意:

  • Rendering Tree 渲染樹并不等同于DOM樹,因為一些像Header或display:none的東西就沒必要放在渲染樹中了。
  • CSS 的 Rule Tree主要是為了完成匹配并把CSS Rule附加上Rendering Tree上的每個Element。也就是DOM結(jié)點。也就是所謂的Frame。
  • 然后,計算每個Frame(也就是每個Element)的位置,這又叫l(wèi)ayout和reflow過程。

3)最后通過調(diào)用操作系統(tǒng)Native GUI的API繪制。

DOM解析

HTML的DOM Tree解析如下:

  1. <html> 
  2. <html> 
  3. <head> 
  4.     <title>Web page parsing</title> 
  5. </head> 
  6. <body> 
  7.     <div> 
  8.         <h1>Web page parsing</h1> 
  9.         <p>This is an example Web page.</p> 
  10.     </div> 
  11. </body> 
  12. </html> 

上面這段HTML會解析成這樣:

下面是另一個有SVG標(biāo)簽的情況。

CSS解析

CSS的解析大概是下面這個樣子(下面主要說的是Gecko也就是Firefox的玩法),假設(shè)我們有下面的HTML文檔:

  1. <doc> 
  2. <title>A few quotes</title> 
  3. <para> 
  4.   Franklin said that <quote>"A penny saved is a penny earned."</quote> 
  5. </para> 
  6. <para> 
  7.   FDR said <quote>"We have nothing to fear but <span>fear itself.</span>"</quote> 
  8. </para> 
  9. </doc> 

于是DOM Tree是這個樣子:

然后我們的CSS文檔是這樣的:

  1. /* rule 1 */ doc { displayblocktext-indent1em; } 
  2. /* rule 2 */ title { displayblockfont-size3em; } 
  3. /* rule 3 */ para { displayblock; } 
  4. /* rule 4 */ [class="emph"] { font-styleitalic; } 

于是我們的CSS Rule Tree會是這個樣子:

注意,圖中的第4條規(guī)則出現(xiàn)了兩次,一次是獨立的,一次是在規(guī)則3的子結(jié)點。所以,我們可以知道,建立CSS Rule Tree是需要比照著DOM Tree來的。CSS匹配DOM Tree主要是從右到左解析CSS的Selector,好多人以為這個事會比較快,其實并不一定。關(guān)鍵還看我們的CSS的Selector怎么寫了。

注意:CSS匹配HTML元素是一個相當(dāng)復(fù)雜和有性能問題的事情。所以,你就會在N多地方看到很多人都告訴你,DOM樹要小,CSS盡量用id和class,千萬不要過渡層疊下去,……

#p#

通過這兩個樹,我們可以得到一個叫Style Context Tree,也就是下面這樣(把CSS Rule結(jié)點Attach到DOM Tree上):

所以,F(xiàn)irefox基本上來說是通過CSS 解析 生成 CSS Rule Tree,然后,通過比對DOM生成Style Context Tree,然后Firefox通過把Style Context Tree和其Render Tree(Frame Tree)關(guān)聯(lián)上,就完成了。注意:Render Tree會把一些不可見的結(jié)點去除掉。而Firefox中所謂的Frame就是一個DOM結(jié)點,不要被其名字所迷惑了。

注:Webkit不像Firefox要用兩個樹來干這個,Webkit也有Style對象,它直接把這個Style對象存在了相應(yīng)的DOM結(jié)點上了。

渲染

渲染的流程基本上如下(黃色的四個步驟):

計算CSS樣式

構(gòu)建Render Tree

Layout – 定位坐標(biāo)和大小,是否換行,各種position, overflow, z-index屬性 ……

正式開畫

注意:上圖流程中有很多連接線,這表示了Javascript動態(tài)修改了DOM屬性或是CSS屬會導(dǎo)致重新Layout,有些改變不會,就是那些指到天上的箭頭,比如,修改后的CSS rule沒有被匹配到,等。

這里重要要說兩個概念,一個是Reflow,另一個是Repaint。這兩個不是一回事。

  • Repaint——屏幕的一部分要重畫,比如某個CSS的背景色變了。但是元素的幾何尺寸沒有變。
  • Reflow——意味著元件的幾何尺寸變了,我們需要重新驗證并計算Render Tree。是Render Tree的一部分或全部發(fā)生了變化。這就是Reflow,或是Layout。(HTML使用的是flow based layout,也就是流式布局,所以,如果某元件的幾何尺寸發(fā)生了變化,需要重新布局,也就叫reflow)reflow 會從<html>這個root frame開始遞歸往下,依次計算所有的結(jié)點幾何尺寸和位置,在reflow過程中,可能會增加一些frame,比如一個文本字符串必需被包裝起來。

下面是一個打開Wikipedia時的Layout/reflow的視頻(注:HTML在初始化的時候也會做一次reflow,叫 intial reflow),你可以感受一下:

 

Reflow的成本比Repaint的成本高得多的多。DOM Tree里的每個結(jié)點都會有reflow方法,一個結(jié)點的reflow很有可能導(dǎo)致子結(jié)點,甚至父點以及同級結(jié)點的reflow。在一些高性能的電腦上也許還沒什么,但是如果reflow發(fā)生在手機上,那么這個過程是非常痛苦和耗電的。

 

所以,下面這些動作有很大可能會是成本比較高的。

  • 當(dāng)你增加、刪除、修改DOM結(jié)點時,會導(dǎo)致Reflow或Repaint
  • 當(dāng)你移動DOM的位置,或是搞個動畫的時候。
  • 當(dāng)你修改CSS樣式的時候。
  • 當(dāng)你Resize窗口的時候(移動端沒有這個問題),或是滾動的時候。
  • 當(dāng)你修改網(wǎng)頁的默認(rèn)字體時。

注:display:none會觸發(fā)reflow,而visibility:hidden只會觸發(fā)repaint,因為沒有發(fā)現(xiàn)位置變化。

多說兩句關(guān)于滾屏的事,通常來說,如果在滾屏的時候,我們的頁面上的所有的像素都會跟著滾 動,那么性能上沒什么問題,因為我們的顯卡對于這種把全屏像素往上往下移的算法是很快。但是如果你有一個fixed的背景圖,或是有些Element不跟 著滾動,有些Elment是動畫,那么這個滾動的動作對于瀏覽器來說會是相當(dāng)相當(dāng)痛苦的一個過程。你可以看到很多這樣的網(wǎng)頁在滾動的時候性能有多差。因為 滾屏也有可能會造成reflow。

基本上來說,reflow有如下的幾個原因:

  • Initial。網(wǎng)頁初始化的時候。
  • Incremental。一些Javascript在操作DOM Tree時。
  • Resize。其些元件的尺寸變了。
  • StyleChange。如果CSS的屬性發(fā)生變化了。
  • Dirty。幾個Incremental的reflow發(fā)生在同一個frame的子樹上。

好了,我們來看一個示例吧:

  1. var bstyle = document.body.style; // cache 
  2.   
  3. bstyle.padding = "20px"// reflow, repaint 
  4. bstyle.border = "10px solid red"//  再一次的 reflow 和 repaint 
  5.   
  6. bstyle.color = "blue"// repaint 
  7. bstyle.backgroundColor = "#fad"// repaint 
  8.   
  9. bstyle.fontSize = "2em"// reflow, repaint 
  10.   
  11. // new DOM element - reflow, repaint 
  12. document.body.appendChild(document.createTextNode('dude!')); 

當(dāng)然,我們的瀏覽器是聰明的,它不會像上面那樣,你每改一次樣式,它就reflow或repaint一次。一般來說,瀏覽器會把這樣的操作積攢一批,然后做一次reflow,這又叫異步reflow或增量異步reflow。但是有些情況瀏覽器是不會這么做的,比如:resize窗口,改變了頁面默認(rèn)的字體,等。對于這些操作,瀏覽器會馬上進行reflow。

#p#

但是有些時候,我們的腳本會阻止瀏覽器這么干,比如:如果我們請求下面的一些DOM值:

  1. offsetTop, offsetLeft, offsetWidth, offsetHeight
  2. scrollTop/Left/Width/Height
  3. clientTop/Left/Width/Height
  4. IE中的 getComputedStyle(), 或 currentStyle

因為,如果我們的程序需要這些值,那么瀏覽器需要返回最新的值,而這樣一樣會flush出去一些樣式的改變,從而造成頻繁的reflow/repaint。

減少reflow/repaint

下面是一些Best Practices:

1)不要一條一條地修改DOM的樣式。與其這樣,還不如預(yù)先定義好css的class,然后修改DOM的className。

  1. // bad 
  2. var left = 10
  3. top = 10
  4. el.style.left = left + "px"
  5. el.style.top  = top  + "px"
  6.   
  7. // Good 
  8. el.className += " theclassname"
  9.   
  10. // Good 
  11. el.style.cssText += "; left: " + left + "px; top: " + top + "px;"

2)把DOM離線后修改。如:

  • 使用documentFragment 對象在內(nèi)存里操作DOM
  • 先把DOM給display:none(有一次reflow),然后你想怎么改就怎么改。比如修改100次,然后再把他顯示出來。
  • clone一個DOM結(jié)點到內(nèi)存里,然后想怎么改就怎么改,改完后,和在線的那個的交換一下。

3)不要把DOM結(jié)點的屬性值放在一個循環(huán)里當(dāng)成循環(huán)里的變量。不然這會導(dǎo)致大量地讀寫這個結(jié)點的屬性。

4)盡可能的修改層級比較低的DOM。當(dāng)然,改變層級比較底的DOM有可能會造成大面積的reflow,但是也可能影響范圍很小。

5)為動畫的HTML元件使用fixed或absoult的position,那么修改他們的CSS是不會reflow的。

6)千萬不要使用table布局。因為可能很小的一個小改動會造成整個table的重新布局。

幾個工具和幾篇文章

有時候,你會也許會發(fā)現(xiàn)在IE下,你不知道你修改了什么東西,結(jié)果CPU一下子就上去了到100%,然后過了好幾秒鐘repaint/reflow才完成,這種事情以IE的年代時經(jīng)常發(fā)生。所以,我們需要一些工具幫我們看看我們的代碼里有沒有什么不合適的東西。

  • Chrome下,Google的SpeedTracer是個非常強悍的工作讓你看看你的瀏覽渲染的成本有多大。其實Safari和Chrome都可以使用開發(fā)者工具里的一個Timeline的東東。
  • Firefox下這個基于Firebug的叫Firebug Paint Events的插件也不錯。
  • IE下你可以用一個叫dynaTrace的IE擴展。

最后,別忘了下面這幾篇提高瀏覽器性能的文章:

原文鏈接:http://coolshell.cn/articles/9666.html

 

 

 

責(zé)任編輯:陳四芳 來源: 酷殼網(wǎng)
相關(guān)推薦

2013-05-23 16:01:56

瀏覽器

2020-11-06 15:20:45

瀏覽器前端架構(gòu)

2022-08-30 09:01:11

瀏覽器渲染前端

2013-11-18 14:42:53

瀏覽器渲染

2012-03-20 11:35:32

傲游手機瀏覽器

2017-03-08 08:31:48

瀏覽器渲染路徑

2013-11-20 10:47:57

瀏覽器渲染html

2017-03-12 10:15:18

瀏覽器DOM樹CSSOM樹

2013-11-20 13:04:41

css瀏覽器渲染

2013-11-20 13:47:43

瀏覽器渲染引擎

2018-01-19 14:39:53

瀏覽器頁面優(yōu)化

2017-10-09 13:39:26

瀏覽器渲染服務(wù)器

2019-01-03 13:09:58

瀏覽器緩存原理

2013-11-18 15:09:34

瀏覽器渲染速度

2022-02-07 21:49:06

瀏覽器渲染chromium

2021-04-19 11:40:15

瀏覽器路徑

2020-03-12 11:29:51

JavaScript瀏覽器語言

2015-02-28 09:39:24

Windows 10Spartan

2021-04-01 06:23:24

CSS33D3D Web 動畫

2019-04-08 10:27:00

渲染瀏覽器DOM
點贊
收藏

51CTO技術(shù)棧公眾號