實例分析瀏覽器中“JavaScript解析器”的工作原理
瀏覽器在讀取HTML文件的時候,只有當遇到<script>標簽的時候,才會喚醒所謂的“JavaScript解析器”開始工作。
JavaScript解析器工作步驟:
1、“找一些東西”: var、 function、 參數(shù);(也被稱之為預解析)
備注:如果遇到重名分為以下兩種情況:
- 遇到變量和函數(shù)重名了,只留下函數(shù)
- 遇到函數(shù)重名了,根據(jù)代碼的上下文順序,留下最后一個
2、逐行解讀代碼。
備注:表達式可以修改預解析的值
JS解析器在執(zhí)行第一步預解析的時候,會從代碼的開始搜索直到結(jié)尾,只去查找var、function和參數(shù)等內(nèi)容。一般把第一步稱之為“JavaScript的預解析”。而且,當找到這些內(nèi)容時,所有的變量,在正式運行代碼之前,都提前賦了一個值:未定義;所有的函數(shù),在正式運行代碼之前,都是整個函數(shù)塊。
實例分析:
實例一:
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>Document</title> <script type="text/javascript"> alert(a); </script> </head> <body> </body> </html>
這段代碼運行后,瀏覽器會報錯。
原因:由于,“JavaScript解析器”在解析JS代碼時,未找到var、function、參數(shù)等其中的任何一個,所以,當逐行執(zhí)行代碼時,因為在“倉庫”中找不到a,不認識a,就會報錯。
實例二:
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>Document</title> <script type="text/javascript"> alert(a); var a = 1; </script> </head> <body> </body> </html>
這段代碼運行后,瀏覽器會彈出”undefined”。
原因:由于,“JavaScript解析器”在解析JS代碼時,找到var關(guān)鍵字,然后得知有一個變量a,所以會給a默認賦值一個undefined值,存入“倉庫”中,所以,當逐行執(zhí)行代碼時,找到變量a,因為此時的a的值為undefined,所以彈出的值為undefined。
實例三:
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>Document</title> <script type="text/javascript"> var a = 1; alert(a); </script> </head> <body> </body> </html>
這段代碼運行后,瀏覽器會彈出數(shù)字“1”。
原因:由于,“JavaScript解析器”在解析JS代碼時,找到var關(guān)鍵字,然后得知有一個變量a,所以會給a默認賦值一個undefined值,存入“倉庫”中,在逐行執(zhí)行代碼時,先找到變量a,此時的a的值為undefined,當執(zhí)行到var a= 1 這行代碼時,a得到了一個新的賦值“1”所以彈出的值為數(shù)字“1”。
實例四:
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>Document</title> <script type="text/javascript"> alert(a); var a = 1; alert(a); function a () { alert(2); } alert(a); var a = 3; alert(a); function a () { alert(4); } alert(a); </script> </head> <body> </body> </html>
這段代碼運行后,彈出值的順序依次為:function a () {alert(4);}、1、1、3、3
原因:由于,“JavaScript解析器”在逐行解析代碼時,先找到var關(guān)鍵字,然后得知有一個變量a,所以會給a默認賦值一個undefined值,存入“倉庫”中,然后,繼續(xù)向下解析代碼,當找到function a () {alert(2);}時,根據(jù)“函數(shù)和變量重名,保留函數(shù)”的規(guī)則,此時的a變?yōu)閒unction () {alert(2);},再繼續(xù)向下找,當找到變量a的時候,不變,仍繼續(xù)向下找,當找到函數(shù)function () {alert(4);}時,根據(jù)“函數(shù)重名上下文”原則,替換為function () {alert(4);},最終a被賦值為function () {alert(4);},存到“倉庫”中,當逐行執(zhí)行代碼時,執(zhí)行到第一個alert(a)時,將會彈出“function () {alert(4);}”,接著向下執(zhí)行,當執(zhí)行到a=1時,由于表達式可以改變預解析的值,所以此時的a變?yōu)?,執(zhí)行到第二個alert(a)時,彈出值為1,當執(zhí)行到function a () {alert(2);}時,由于此時是一個函數(shù)聲明,并不會修改a的值,所以執(zhí)行到第三個alert(a)時,彈出的仍為數(shù)字“1”,依次類推,此后將會陸續(xù)彈出“3”、“3”。
以上就是本文的全部內(nèi)容,希望本文的內(nèi)容對大家的學習或者工作能帶來一定的幫助,同時也希望多多支持腳本之家!
相關(guān)文章
一文詳解JavaScript中的replace()函數(shù)
replace方法的語法是stringObj.replace(rgExp, replaceText),其中stringObj是字符串(string),下面這篇文章主要給大家介紹了關(guān)于JavaScript中replace()函數(shù)的相關(guān)資料,需要的朋友可以參考下2023-01-01JavaScript onclick與addEventListener使用的區(qū)別介紹
addEventListener()方法用于向指定元素添加事件句柄,使用 removeEventListener()方法來移除,onclick和addEventListener事件區(qū)別是:onclick事件會被覆蓋,而addEventListener可以先后運行不會被覆蓋,addEventListener可以監(jiān)聽多個事件2022-09-09在TypeScript項目中搭配Axios封裝后端接口調(diào)用
這篇文章主要介紹了在TypeScript項目中搭配Axios封裝后端接口調(diào)用,本文記錄一下在?TypeScript?項目里封裝?axios?的過程,之前在開發(fā)?StarBlog-Admin?的時候已經(jīng)做了一次封裝,不過那時是JavaScript跟TypeScript還是有些區(qū)別的,需要的朋友可以參考下2024-01-01contains和compareDocumentPosition 方法來確定是否HTML節(jié)點間的關(guān)系
一個很棒的 blog 文章,是 PPK 兩年前寫的,文章中解釋了 contains() 和 compareDocumentPosition() 方法運行在他們各自的瀏覽器上。2011-09-09