成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

瀏覽器如何解析html、css、js

Awbeci / 765人閱讀

摘要:全局變量局部變量全局函數(shù)一段也是一塊域。此時(shí)打印的自然是,要記住相當(dāng)于,所以這時(shí)候改變的是局部變量,并沒有影響到全局變量,所以第二次打印的依然是。

在熟悉了瀏覽器的工作原理之后,今天我們來講講瀏覽器在從服務(wù)器獲取到網(wǎng)頁(yè)文件之后是如何解析的。了解了這個(gè)基礎(chǔ)知識(shí),對(duì)敲出來的代碼,質(zhì)量會(huì)有不小的提升。

一、瀏覽器如何解析html

html文件在沒有寫入html標(biāo)簽之前和txt文本是一個(gè)性質(zhì)的,不含任何樣式。只是單純的文本預(yù)覽文件。一旦加入了html標(biāo)簽,表示內(nèi)容有了語(yǔ)義!瀏覽器的渲染引擎才會(huì)根據(jù)標(biāo)簽的語(yǔ)義開始解析。

我們現(xiàn)在所看到的html原本分為html和xhtml兩個(gè)版本,它們的區(qū)別是xhtml比html更為嚴(yán)格,規(guī)范性更強(qiáng)。由于html比xhtml更加“寬松”,使網(wǎng)頁(yè)作者的生活變得輕松。所以這使得html很流行。

渲染引擎的基本工作流程

解析HTML構(gòu)建DOM樹

渲染樹構(gòu)建

渲染樹布局

繪制渲染樹

渲染引擎會(huì)解析HTML文檔并把標(biāo)簽轉(zhuǎn)換成內(nèi)容樹中的DOM節(jié)點(diǎn)。它會(huì)解析style元素和外部文件中的樣式數(shù)據(jù)。樣式數(shù)據(jù)和HTML中的顯示控制將共同用來創(chuàng)建另一棵樹——渲染樹。渲染引擎會(huì)嘗試盡快的把內(nèi)容顯示出來。它不會(huì)等到所有HTML都被解析完才創(chuàng)建并布局渲染樹。它會(huì) 在處理后續(xù)內(nèi)容的同時(shí)把處理過的局部?jī)?nèi)容先展示出來。

不同瀏覽器使用的內(nèi)核也許不同,但是整個(gè)渲染流程大同小異。

開始解析

解析一個(gè)文檔意味著把它翻譯成有意義的結(jié)構(gòu)以供代碼使用。解析的結(jié)果通常是一個(gè)表征文檔的由節(jié)點(diǎn)組成的樹,稱為解析樹或句法樹。
解析器通常把工作分給兩個(gè)組件——分詞程序負(fù)責(zé)把輸入切分成合法符號(hào)序列,解析程序負(fù)責(zé)按照句法規(guī)則分析文檔結(jié)構(gòu)和構(gòu)建句法樹。詞法分析器知道如何過濾像空格,換行之類的無關(guān)字符。
解析器輸出的樹是由DOM元素和屬性節(jié)點(diǎn)組成的。DOM的全稱為:Document Object Model。它是HTML文檔的對(duì)象化描述,也是HTML元素與外界(如Javascript)的接口。

DOM與標(biāo)簽幾乎有著一一對(duì)應(yīng)的關(guān)系,如下面的標(biāo)簽


    
        

Hello World

會(huì)被轉(zhuǎn)換成如的DOM樹:

我們都知道代碼是逐行執(zhí)行的,解析也是如此。這里涉及到一個(gè)解析算法,算法太復(fù)雜,簡(jiǎn)單的理解為:解析由兩部分組成:分詞與構(gòu)建樹。它把輸入解析成符號(hào)序列。在HTML中符號(hào)就是開始標(biāo)簽,結(jié)束標(biāo)簽,屬性名稱和屬生值。分詞器識(shí)別這些符號(hào)并將其送入樹構(gòu)建者,然后繼續(xù)分析處理下一個(gè)符號(hào),直到輸入結(jié)束。

瀏覽器的容錯(cuò)機(jī)制




Really lousy HTML

像這段代碼很明顯不符合規(guī)范,盡管如此,瀏覽器還是在解析的過程中修復(fù)了html作者的錯(cuò)誤內(nèi)容并繼續(xù)工作。具體是怎么修復(fù)的,咱不做深入了解。要保證的是我們?cè)谇么a的時(shí)候一定要按照規(guī)范來,盡量少給瀏覽器添堵。

二、瀏覽器如何解析css

這里我主要講一下css解析選擇器的匹配規(guī)則,我們都知道css的選擇器都是全局的。這樣有好也有壞!好處是代碼重用率高、可以把css文件合并、拆分做的像硬件一樣。壞處是css寫法特別的靈活,也因?yàn)殪`活,所以容易耦合在一起。

實(shí)際上CSS選擇器的讀取順序是從右向左。

#molly div.haha span{color:#f00}

如上面的代碼,瀏覽器會(huì)按照從右向左的順序去讀取選擇器。先找到span然后順著往上找到class為“haha”的div再找到id為“molly”的元素。成功匹配到則加入結(jié)果集,如果直到根元素html都沒有匹配,則不再遍歷這條路徑,從下一個(gè)span開始重復(fù)這個(gè)過程。整個(gè)過程會(huì)形成一條符合規(guī)則的索引樹,樹由上至下的節(jié)點(diǎn)是規(guī)則中從右向左的一個(gè)個(gè)選擇符匹配的節(jié)點(diǎn)。

如果從左向右的順序讀取,在執(zhí)行到左邊的分支后發(fā)現(xiàn)沒有相對(duì)應(yīng)標(biāo)簽匹配,則會(huì)回溯到上一個(gè)節(jié)點(diǎn)再繼續(xù)遍歷,直到找到或者沒有相匹配的標(biāo)簽才結(jié)束。如果有100個(gè)甚至1000個(gè)分支的時(shí)候會(huì)消耗很多性能。反之從右向左查找極大的縮小的查找范圍從而提高了性能。這就解釋了為什么id選擇器大于類選擇器,類選擇器大于元素選擇器。

三、瀏覽器如何解析js

在瀏覽器中有一個(gè)“js解析器”的工具,專門用來解析我們的js代碼。在這里我們只需要關(guān)注解析的其中兩個(gè)步驟就行了,其它的不做研究。

js預(yù)解析

逐行解析代碼

當(dāng)瀏覽器遇到j(luò)s代碼時(shí),立馬召喚“js解析器”出來工作。這個(gè)時(shí)候還不慌,得先做好準(zhǔn)備工作。解析器會(huì)找到j(luò)s當(dāng)中的所有變量、函數(shù)、參數(shù)等等一大堆。并且把變量賦值為未定義(undefeated),把函數(shù)取出來成為一個(gè)函數(shù)塊,然后存放到倉(cāng)庫(kù)當(dāng)中。這件事情做完了之后才開始逐行解析代碼(由上向下,由左向右),然后再去和倉(cāng)庫(kù)進(jìn)行匹配。



再看一下這段代碼

在js預(yù)解析的時(shí)候,在遇到變量和函數(shù)重名的時(shí)候,只會(huì)保留函數(shù)塊。在逐行解析代碼的時(shí)候表達(dá)式(+、-、*、/、%、++、–、 參數(shù) ……)會(huì)改變倉(cāng)庫(kù)里對(duì)應(yīng)的值。

來!繼續(xù)深入…
我們來了解一個(gè)詞“作用域”,現(xiàn)在把這個(gè)詞拆分一下。
作用:讀、寫操作
域:空間、范圍、區(qū)域…
連起來就是能夠進(jìn)行讀寫操作的一個(gè)區(qū)域。
“域”:函數(shù)、json、……都是作為一塊作用域。
全局變量、局部變量、全局函數(shù)
一段 也是一塊域。在域解析的時(shí)候,也是由上向下開始解析。這就解釋了為什么引用的外部公共js文件(比如:jquery)應(yīng)該放到自定義js上邊的原因。

再來看一下這段代碼

繼續(xù)跟蹤一下解析器的解析過程:首先函數(shù)fn()外部的a是一個(gè)全局變量,fn()里面的a是一個(gè)局部變量。fn()函數(shù)同時(shí)是一個(gè)作用域,只要是作用域,就得做預(yù)解析和逐行解析的步驟。所以第一個(gè)alert打印的是fn()作用域的倉(cāng)庫(kù)指向的變量a,即為undefeated。第二個(gè)alert打印的是全局的變量a,即為1。

接下來繼續(xù)看代碼,基本雷同的代碼,我改變其中一小個(gè)地方。

看到這里當(dāng)解析到fn()的時(shí)候,發(fā)現(xiàn)里面并沒有任何變量,所以也就不往倉(cāng)庫(kù)里面存什么,此時(shí)的倉(cāng)庫(kù)里面是空的,啥也沒有。但是這個(gè)時(shí)候解析并沒有結(jié)束,而是從函數(shù)里面向外開始找,找到全局的變量a。此時(shí)打印的正式全局變量a的值。

這里就涉及到一個(gè)作用域鏈的問題。整個(gè)解析過程像是一條鏈子一樣。由上向下,由里到外!局部能夠讀寫全局,全局無法讀寫局部。

來,繼續(xù)看代碼,基本雷同的代碼,我再次改變其中一小個(gè)地方。

千萬不能忘了,在預(yù)解析的時(shí)候?yàn)g覽器除了要找變量和函數(shù)之外還需要找一些參數(shù),并且賦值為未定義。所以這里的fn(a)相當(dāng)于fn(var a),這個(gè)時(shí)候的邏輯就和第一段實(shí)例代碼一樣了。

繼續(xù)搞事情,繼續(xù)看代碼,基本雷同的代碼,我再次改變其中一小個(gè)地方。

當(dāng)代碼執(zhí)行到fn(a);的時(shí)候調(diào)用的fn()函數(shù)并且把全局變量a作為參數(shù)傳遞進(jìn)去。此時(shí)打印的自然是1,要記住function fn(a)相當(dāng)于function fn(var a),所以這時(shí)候a=2;改變的是局部變量a,并沒有影響到全局變量a,所以第二次打印的依然是1。

對(duì)于瀏覽器如何解析html、css、js就先介紹到這兒了,如有不對(duì)的地方,歡迎指正。

我是貓哆哩,一個(gè)不成熟的程序員!

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/54587.html

相關(guān)文章

  • 覽器如何解析html、cssjs

    摘要:全局變量局部變量全局函數(shù)一段也是一塊域。此時(shí)打印的自然是,要記住相當(dāng)于,所以這時(shí)候改變的是局部變量,并沒有影響到全局變量,所以第二次打印的依然是。 在熟悉了瀏覽器的工作原理之后,今天我們來講講瀏覽器在從服務(wù)器獲取到網(wǎng)頁(yè)文件之后是如何解析的。了解了這個(gè)基礎(chǔ)知識(shí),對(duì)敲出來的代碼,質(zhì)量會(huì)有不小的提升。 一、瀏覽器如何解析html html文件在沒有寫入html標(biāo)簽之前和txt文本是一個(gè)性質(zhì)的...

    Panda 評(píng)論0 收藏0
  • 覽器如何解析html、css、js

    摘要:全局變量局部變量全局函數(shù)一段也是一塊域。此時(shí)打印的自然是,要記住相當(dāng)于,所以這時(shí)候改變的是局部變量,并沒有影響到全局變量,所以第二次打印的依然是。 在熟悉了瀏覽器的工作原理之后,今天我們來講講瀏覽器在從服務(wù)器獲取到網(wǎng)頁(yè)文件之后是如何解析的。了解了這個(gè)基礎(chǔ)知識(shí),對(duì)敲出來的代碼,質(zhì)量會(huì)有不小的提升。 一、瀏覽器如何解析html html文件在沒有寫入html標(biāo)簽之前和txt文本是一個(gè)性質(zhì)的...

    Salamander 評(píng)論0 收藏0
  • 切圖崽的自我修養(yǎng)-提高項(xiàng)目加載速度

    摘要:前言我的項(xiàng)目沒問題,是用戶的網(wǎng)絡(luò)環(huán)境不夠好前端作為一個(gè)最貼近用戶的技術(shù)工種,毫無疑問應(yīng)該把戶體驗(yàn)放在第一位而用戶體驗(yàn),基本正比于頁(yè)面的打開速度,特別是做移動(dòng)端的同學(xué),所以如何優(yōu)化自己的項(xiàng)目,提高頁(yè)面的加載速度成為重中之重資源的下載及解析對(duì)前 前言 我的項(xiàng)目沒問題,是用戶的網(wǎng)絡(luò)環(huán)境不夠好 前端作為一個(gè)最貼近用戶的技術(shù)工種,毫無疑問應(yīng)該把戶體驗(yàn)放在第一位. 而用戶體驗(yàn),基本正比于頁(yè)面的打開...

    yanwei 評(píng)論0 收藏0
  • css、js阻塞

    摘要:例如,當(dāng)解析器被腳本阻塞時(shí),解析器雖然會(huì)停止構(gòu)建,但仍會(huì)識(shí)別該腳本后面的資源,并進(jìn)行預(yù)加載。也就是說,會(huì)阻塞頁(yè)面的渲染但是,并不會(huì)阻塞的解析。的加載不會(huì)阻塞頁(yè)面的渲染和資源的加載,一旦加載到就會(huì)立刻執(zhí)行。 大家是不是會(huì)遇到這樣的一個(gè)問題,頁(yè)面加載速度過慢,瀏覽器老在轉(zhuǎn)圈圈,頁(yè)面部分內(nèi)容需要花費(fèi)較多的時(shí)間才能加載出來? 要明白上述問題,我們需要知道是什么在阻塞頁(yè)面的渲染? 1、瀏覽器如何...

    gxyz 評(píng)論0 收藏0
  • webpack的配置和安裝(2)

    摘要:是用來做瀏覽器兼容的,這個(gè)必須要配置,否則在打包的時(shí)候就會(huì)報(bào)錯(cuò),在這個(gè)中我們引用了之前安裝的,這個(gè)是用來自動(dòng)匹配瀏覽器來補(bǔ)全前綴的,然后我們?cè)O(shè)置為最近五個(gè)瀏覽器版本。 上一篇文章講述了如何安裝和配置webpack的基礎(chǔ)依賴,可以看鏈接描述 這篇文章我們來看看如何配置webpack最重要的配置文件webpack.config.js 首先我們先在根目錄下新建一個(gè)webpack.config...

    chadLi 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<