溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶(hù)服務(wù)條款》

HTML和CSS及JS是如何變成頁(yè)面的

發(fā)布時(shí)間:2021-10-12 16:45:31 來(lái)源:億速云 閱讀:139 作者:柒染 欄目:web開(kāi)發(fā)

本篇文章給大家分享的是有關(guān)HTML和CSS及JS是如何變成頁(yè)面的,小編覺(jué)得挺實(shí)用的,因此分享給大家學(xué)習(xí),希望大家閱讀完這篇文章后可以有所收獲,話(huà)不多說(shuō),跟著小編一起來(lái)看看吧。

我們經(jīng)常寫(xiě) HTML 、 CSS 和 JavaScript ,寫(xiě)好這些之后,我們就會(huì)在瀏覽器中看到頁(yè)面,那瀏覽器究竟在這背后做了一些什么事情呢?本篇文章將揭曉答案!

了解瀏覽器的渲染原理是我們?cè)谕ㄍ顚哟蔚那岸碎_(kāi)發(fā)中不可缺少的,它可以讓我們從更深層次、角度去考慮性能優(yōu)化等~

下面進(jìn)入正文~

進(jìn)程、線(xiàn)程

瀏覽器會(huì)分配一個(gè)線(xiàn)程“自上而下,從左到右”依次解析和渲染代碼,那么進(jìn)程和線(xiàn)程是什么,它們之間有著怎樣的關(guān)系呢?

進(jìn)程

一個(gè)進(jìn)程就是一個(gè)程序運(yùn)行的實(shí)例。當(dāng)啟動(dòng)一個(gè)程序的時(shí)候,操作系統(tǒng)會(huì)為該程序創(chuàng)建一塊內(nèi)存,用來(lái)存放代碼,運(yùn)行中的數(shù)據(jù)和一個(gè)執(zhí)行任務(wù)的主線(xiàn)程,這樣的一個(gè)運(yùn)行環(huán)境就叫進(jìn)程

線(xiàn)程

線(xiàn)程不能單獨(dú)存在,它是由進(jìn)程來(lái)啟動(dòng)和管理的。線(xiàn)程依附于進(jìn)程,進(jìn)程中使用多線(xiàn)程并行處理能提升運(yùn)算效率

兩者之間的關(guān)系

1、進(jìn)程中的任意一線(xiàn)程執(zhí)行出錯(cuò),都會(huì)導(dǎo)致整個(gè)進(jìn)程的崩潰

2、線(xiàn)程之間可以共享數(shù)據(jù)

3、當(dāng)一個(gè)進(jìn)程關(guān)閉后,操作系統(tǒng)會(huì)回收進(jìn)程所占用的內(nèi)存

4、進(jìn)程之間的內(nèi)容相互隔離

渲染機(jī)制

從HTML、CSS和JavaScript開(kāi)始

了解瀏覽器的渲染原理,我們就要從理解 HTML 、 CSS 和 JavaScrip 開(kāi)始,我們先來(lái)看一張圖

HTML和CSS及JS是如何變成頁(yè)面的

HTML (超文本標(biāo)記語(yǔ)言),顧名思義,由標(biāo)記(標(biāo)簽)和文本組成,每個(gè)標(biāo)簽都有自己的語(yǔ)意,瀏覽器會(huì)根據(jù)標(biāo)簽和文本展示對(duì)應(yīng)的內(nèi)容。

CSS (層疊樣式表),由選擇器和屬性組成,它可以改變 HTML 的樣式,比如上圖中,我們改變了 span 的顏色由藍(lán)色為綠色。

JavaScript ,我們可以通過(guò) JS 完成很多事情,例如上圖中修改樣式。

下面開(kāi)始分析渲染的原理

渲染流水線(xiàn)

渲染模塊由于渲染的機(jī)制的復(fù)雜,被劃分為了很多子階段,輸入的 HTML 經(jīng)過(guò)這些子階段,最后會(huì)輸出為像素。這樣的處理流程就叫做 渲染流水線(xiàn)

HTML和CSS及JS是如何變成頁(yè)面的

按照渲染的時(shí)間順序,流水線(xiàn)可分為幾個(gè)子階段:構(gòu)建 DOM 樹(shù)、樣式計(jì)算、布局階段、分層、繪制、分塊、光柵化和合成

構(gòu)建DOM樹(shù)

由于瀏覽器無(wú)法直接理解和使用 HTML ,所以需要將 HTML 轉(zhuǎn)換為瀏覽器能夠理解的結(jié)構(gòu)( DOM 樹(shù))

樹(shù)結(jié)構(gòu)示意圖

HTML和CSS及JS是如何變成頁(yè)面的

DOM樹(shù)的構(gòu)建過(guò)程

我們來(lái)分析一下下面這段代碼會(huì)構(gòu)建出一棵什么樣的 DOM 樹(shù)

HTML和CSS及JS是如何變成頁(yè)面的

我們先將上面的代碼運(yùn)行,然后在瀏覽器控制臺(tái)輸入 document ,看看會(huì)有什么效果

HTML和CSS及JS是如何變成頁(yè)面的

我們一層級(jí)一層級(jí)的打開(kāi)就會(huì)看到如上圖的效果,我們可以根據(jù)這每一層級(jí)展開(kāi)的效果,繪制出一棵 DOM 樹(shù)結(jié)構(gòu),如下:

HTML和CSS及JS是如何變成頁(yè)面的

接下來(lái),我們?cè)囈幌吕?JS 修改一下內(nèi)容,看有什么改變:

HTML和CSS及JS是如何變成頁(yè)面的

我們可以看到“瀏覽器”的文字變成了“chrome”

HTML和CSS及JS是如何變成頁(yè)面的

再來(lái)看一下 DOM 樹(shù)是否有改變

HTML和CSS及JS是如何變成頁(yè)面的

我們看到在“瀏覽器”的位置換成了“chrome”,那么如何讓 DOM 節(jié)點(diǎn)擁有樣式?

樣式計(jì)算

樣式計(jì)算,顧名思義,就是 計(jì)算出 DOM 節(jié)點(diǎn)中每個(gè)元素的具體樣式 ,這個(gè)階段會(huì)分為三部分:

  • 把 CSS 轉(zhuǎn)換為瀏覽器能夠理解的結(jié)構(gòu)

  • 轉(zhuǎn)換樣式表中的屬性值,使其標(biāo)準(zhǔn)化

  • 計(jì)算出 DOM 樹(shù)中每個(gè)節(jié)點(diǎn)的樣式

CSS樣式來(lái)源
  • link 導(dǎo)入外部樣式資源

瀏覽器會(huì)新開(kāi)辟一個(gè)線(xiàn)程,去服務(wù)器獲取對(duì)應(yīng)的資源文件(不阻礙主線(xiàn)程的渲染)

  • style 內(nèi)嵌樣式

從上到下解析,解析完繼續(xù)解析 DOM 結(jié)構(gòu)。在真實(shí)項(xiàng)目中,如果 css 代碼不是很多,或是移動(dòng)端項(xiàng)目,我們應(yīng)該使用內(nèi)嵌式,以此來(lái)減少 http 資源的請(qǐng)求,提高頁(yè)面渲染速度

  • 行內(nèi)樣式

  • @import 導(dǎo)入

它是同步的,不會(huì)開(kāi)辟新線(xiàn)程去加載資源文件,而是讓主線(xiàn)程去獲取,這阻礙 DOM 結(jié)構(gòu)的繼續(xù)渲染;只有把外部樣式導(dǎo)入進(jìn)來(lái),并且解析后,才會(huì)繼續(xù)渲染 DOM 結(jié)構(gòu)

把CSS轉(zhuǎn)換為瀏覽器能夠理解的結(jié)構(gòu)

瀏覽器就像不能理解 HTML 一樣,不理解 CSS ,所以當(dāng)渲染引擎接收到 CSS 文件時(shí),會(huì)執(zhí)行轉(zhuǎn)換操作,將 CSS 文本轉(zhuǎn)換為瀏覽器可以理解的 styleSheets 結(jié)構(gòu)。

在 HTML 中,在瀏覽器中輸入 document 可以查看 html 的結(jié)構(gòu)。在 css 中,可以輸入 document.styleSheets 看到 css 的結(jié)構(gòu)

HTML和CSS及JS是如何變成頁(yè)面的

現(xiàn)在的結(jié)構(gòu)是空的,我們來(lái)加一些樣式,看看效果

HTML和CSS及JS是如何變成頁(yè)面的

轉(zhuǎn)換樣式表中的屬性值,使其標(biāo)準(zhǔn)化

屬性值標(biāo)準(zhǔn)化就是將所有值轉(zhuǎn)換為渲染引擎容易理解的、標(biāo)準(zhǔn)化的計(jì)算值。我們大致看一下效果:

  • 標(biāo)準(zhǔn)化前

body {     font-size: 2em;     color: black;     font-weight: bold;     ... }
  • 標(biāo)準(zhǔn)化后

body {     font-size: 16px;     color: rgb(0, 0, 0);     font-weight: 700;     ... }
計(jì)算出DOM樹(shù)中每個(gè)節(jié)點(diǎn)的具體樣式

樣式計(jì)算有兩個(gè)CSS的規(guī)則:繼承規(guī)則和層疊規(guī)則

  • CSS繼承規(guī)則

CSS 繼承就是每個(gè) DOM 節(jié)點(diǎn)都包含有父節(jié)點(diǎn)的樣式。我們來(lái)看一下下面這段代碼中如何應(yīng)用到 DOM 節(jié)點(diǎn)上

<!DOCTYPE html> <html lang="en"> <head>     <meta charset="UTF-8">     <title>Document</title>     <style>         h2 {             color: red;         }          div {             color: blue;         }          span {             font-size: 16px;         }     </style> </head> <body>     <h2>掘金</h2>     <div>         <span>瀏覽器</span>         <span>渲染原理</span>         構(gòu)建DOM樹(shù)     </div> </body> </html>
子節(jié)點(diǎn)會(huì)擁有父節(jié)點(diǎn)的樣式,由此我們可以畫(huà)出這樣一張圖

HTML和CSS及JS是如何變成頁(yè)面的

我們還可以打開(kāi)控制臺(tái),看一下選中 span 標(biāo)簽,都會(huì)看到哪些內(nèi)容

HTML和CSS及JS是如何變成頁(yè)面的

通過(guò)上圖,我們可看到一個(gè)元素的樣式、繼承過(guò)程等, userAgent 樣式是瀏覽器默認(rèn)的內(nèi)置樣式,如果我們不提供任何樣式,就會(huì)使用此樣式。

  • 樣式層疊規(guī)則

層疊在 CSS 處于核心地位,它是 CSS 的一個(gè)基本特征,它定義了如何合并來(lái)自多個(gè)源的屬性值的算法。

樣式計(jì)算階段最終輸出的內(nèi)容是每個(gè) DOM 節(jié)點(diǎn)的樣式,并且保存在了 ComputedStyle 中。我們可以通過(guò)控制臺(tái)看到某個(gè) DOM 元素最終的計(jì)算樣式

HTML和CSS及JS是如何變成頁(yè)面的

布局階段

現(xiàn)在我們不知道 DOM 元素的幾何位置信息,所以現(xiàn)在我們需要計(jì)算出 DOM 樹(shù)中可見(jiàn)元素的幾何位置,這個(gè)計(jì)算過(guò)程就叫做布局。布局階段有兩個(gè)過(guò)程:

  • 創(chuàng)建布局樹(shù)

  • 布局計(jì)算

創(chuàng)建布局樹(shù)

創(chuàng)建布局樹(shù)的意思就是創(chuàng)建一棵只包含可見(jiàn)元素的樹(shù)。我們來(lái)看下面一段代碼創(chuàng)建布局樹(shù)的過(guò)程

<!DOCTYPE html> <html lang="en"> <head>     <meta charset="UTF-8">     <title>Document</title>     <style>         h2 {             color: red;         }          div {             color: blue;         }          div span {             font-size: 16px;         }          div span:last-child {             display: none;         }     </style> </head> <body>     <h2>掘金</h2>     <div>         <span>瀏覽器</span>         <span>渲染原理</span>         構(gòu)建DOM樹(shù)     </div> </body> </html>
構(gòu)建布局樹(shù)的過(guò)程中,
 DOM 樹(shù)中所有不可見(jiàn)的節(jié)點(diǎn)都不會(huì)包含在這棵樹(shù)中。瀏覽器會(huì)遍歷 DOM 
樹(shù)中所有能看見(jiàn)的節(jié)點(diǎn),然后把這些節(jié)點(diǎn)加入到布局中;不可見(jiàn)的節(jié)點(diǎn)就會(huì)被忽略, head 標(biāo)簽下面的內(nèi)容、 div 下最后一個(gè) span 
節(jié)點(diǎn)都不會(huì)在布局樹(shù)中,我們看一下這個(gè)過(guò)程圖感受一下~

HTML和CSS及JS是如何變成頁(yè)面的

布局計(jì)算

布局計(jì)算就是計(jì)算布局樹(shù)節(jié)點(diǎn)的坐標(biāo)位置。這個(gè)計(jì)算過(guò)程極為復(fù)雜。

分層

渲染引擎會(huì)為特定的節(jié)點(diǎn)生成專(zhuān)用的圖層,并生成一棵對(duì)應(yīng)的圖層樹(shù)。這樣做是因?yàn)轫?yè)面中可能含有很多復(fù)雜的效果,我們可以打開(kāi)控制臺(tái)看一下頁(yè)面的分層情況

HTML和CSS及JS是如何變成頁(yè)面的

HTML和CSS及JS是如何變成頁(yè)面的

我們可以看到,渲染引擎給頁(yè)面分了很多圖層,這些圖層會(huì)按照一定順序疊加在一起,形成最終的頁(yè)面

HTML和CSS及JS是如何變成頁(yè)面的

那么圖層的來(lái)源有哪些?

1、擁有層疊上下文屬性的元素會(huì)被提升為單獨(dú)的一層

層疊上下文可以使能夠使 HTML 元素具有三維的概念,這些 HTML 元素按照自身屬性的優(yōu)先級(jí)分布在垂直于這個(gè)二維平面的 z 軸上。哪些元素具有層疊上下文屬性?

HTML和CSS及JS是如何變成頁(yè)面的

2、需要剪裁的地方會(huì)被創(chuàng)建為圖層

當(dāng)我們創(chuàng)建一個(gè)有寬度和高度的 div 時(shí),里面的文字內(nèi)容可能會(huì)超出這個(gè)區(qū)域,這時(shí)候渲染引擎會(huì)把裁剪文字內(nèi)容的一部分用于顯示在 div 區(qū)域,例如

<!DOCTYPE html> <html lang="en"> <head>     <meta charset="UTF-8">     <meta name="viewport" content="width=device-width, initial-scale=1.0">     <title>Document</title>     <style>         div {             width: 100px;             height: 100px;             background: yellow;             overflow: auto;         }     </style> </head> <body>     <div>         我們經(jīng)常寫(xiě)`HTML`、`CSS`和`JavaScript`,寫(xiě)好這些之后,我們就會(huì)在瀏覽器中看到頁(yè)面,那瀏覽器究竟在這背后做了一些什么事情呢?本篇文章將揭曉答案!          了解瀏覽器的渲染原理是我們?cè)谕ㄍ顚哟蔚那岸碎_(kāi)發(fā)中不可缺少的,它可以讓我們從更深層次、角度去考慮性能優(yōu)化等~     </div> </body> </html>
運(yùn)行結(jié)果

HTML和CSS及JS是如何變成頁(yè)面的

我們?cè)俅蜷_(kāi)控制臺(tái)的 layers 看一下效果

HTML和CSS及JS是如何變成頁(yè)面的

可以看到渲染引擎為文字部分單獨(dú)創(chuàng)建了一個(gè)圖層。

在布局樹(shù)中的節(jié)點(diǎn)如果擁有對(duì)應(yīng)的圖層,這個(gè)節(jié)點(diǎn)就是一個(gè)圖層,如果沒(méi)有,這個(gè)節(jié)點(diǎn)就屬于父節(jié)點(diǎn)的圖層,如下圖:

HTML和CSS及JS是如何變成頁(yè)面的

圖層繪制

創(chuàng)建好圖層樹(shù)后,渲染引擎會(huì)繪制圖層樹(shù)中的每個(gè)圖層。渲染引擎會(huì)將圖層繪制分解為很多小的繪制指令,然后將這些指令按照順序組成待繪制列表,我們可以打開(kāi)控制臺(tái)的 layers ,選擇 document 層,看一下效果

HTML和CSS及JS是如何變成頁(yè)面的

柵格化操作

柵格化就是將圖塊轉(zhuǎn)換位位圖,圖塊是柵格化執(zhí)行的最小單位。渲染進(jìn)程維護(hù)了一個(gè)柵格化的線(xiàn)程池,所有圖塊的柵格化都是在線(xiàn)程池內(nèi)執(zhí)行的。

圖層繪制列表準(zhǔn)備好之后,主線(xiàn)程會(huì)把這個(gè)繪制列表提交給合成線(xiàn)程,繪制操作由渲染引擎中的合成線(xiàn)程來(lái)完成。

合成線(xiàn)程將圖層劃分為圖塊,然后合成線(xiàn)程會(huì)按照視口(可見(jiàn)區(qū)域)附近的圖塊優(yōu)先生成位圖。

合成與顯示

所有的圖塊都被光柵化后,合成線(xiàn)程會(huì)生成一個(gè)繪制圖塊的命令( DrawQuad ),然后將該命令提交給瀏覽器進(jìn)程。瀏覽器進(jìn)程里面 viz 組件用來(lái)接收 DrawQuad 命令,將其頁(yè)面內(nèi)容繪制到內(nèi)存中,最后將內(nèi)存顯示到屏幕。這個(gè)時(shí)候,我們就看到了頁(yè)面

完善渲染流水線(xiàn)示意圖

根據(jù)上文中描述,我們可以畫(huà)出這樣一張圖

HTML和CSS及JS是如何變成頁(yè)面的

我還在網(wǎng)上找到了另外一張圖

HTML和CSS及JS是如何變成頁(yè)面的

這兩張圖都是描述瀏覽器的渲染流程的。

以上就是HTML和CSS及JS是如何變成頁(yè)面的,小編相信有部分知識(shí)點(diǎn)可能是我們?nèi)粘9ぷ鲿?huì)見(jiàn)到或用到的。希望你能通過(guò)這篇文章學(xué)到更多知識(shí)。更多詳情敬請(qǐng)關(guān)注億速云行業(yè)資訊頻道。

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI