更新時間:2020-08-05 來源:黑馬程序員 瀏覽量:
JavaScript是解釋型的語言,但是他并不是真的在運行的時候逐句的往下解析執(zhí)行。
我們來看下面這個例子:
在上面這段代碼中,函數(shù)func的調用是在其聲明之前,如果說JavaScript代碼真的是逐句的解析執(zhí)行,那么在第一句調用的時候就會出錯,然而事實并非如此,上面的代碼可以正常執(zhí)行,并且alert出來`Function has been called`。
所以,可以得出結論,JavaScript并非僅在運行時簡簡單單的逐句解析執(zhí)行!
JavaScript引擎在對JavaScript代碼進行解釋執(zhí)行之前,會對JavaScript代碼進行預解析,在預解析階段,會將以關鍵字`var`和`function`開頭的語句塊提前進行處理。
關鍵問題是怎么處理呢?
當變量和函數(shù)的聲明處在作用域比較靠后的位置的時候,變量和函數(shù)的聲明會被提升到作用域的開頭。
重新來看上面的那段代碼
由于JavaScript的預解析機制,上面的代碼就等效于:
看完函數(shù)聲明的提升,再來看一個變量聲明提升的例子:
由于JavaScript的預解析機制,上面這段代碼,alert出來的值是undefined,如果沒有預解析,代碼應該會直接報錯a is not defined,而不是輸出值。
Wait a minute, 不是說要提前的嗎?那不是應該alert出來1,為什么是undefined?
那么在這里有必要說一下聲明、定義、初始化的區(qū)別。其實這幾個概念是C系語言的人應該都比較了解的。
所以我們說的提升,是聲明的提升。
那么再回過頭看,上面的代碼就等效于:
通過上一小節(jié)的內容,我們對變量、函數(shù)聲明提升已經(jīng)有了一個最基本的理解。那么接下來,我們就來分析一些略復雜的情況。
觀察下面這段代碼:
輸出結果為:
原因分析:由于預解析機制,func1的聲明會被提升,提升之后的代碼為:
同名的函數(shù),后面的會覆蓋前面的,所以兩次輸出結果都是`This is last func1`。
當出現(xiàn)變量聲明和函數(shù)同名的時候,只會對函數(shù)聲明進行提升,變量會被忽略。所以上面的代碼的輸出結果為
我們還是來吧預解析之后的代碼展現(xiàn)出來:
再來看一種
代碼執(zhí)行結果為:
直接上預解析后的代碼:
聲明提升并不是將所有的聲明都提升到window對象下面,提升原則是提升到變量運行的環(huán)境(作用域)中去。
還是直接把預解析之后的代碼寫出來:
分段,其實就分script標簽的
在上面代碼中,第一個script標簽中的兩個`func`進行了提升,第二個`func`覆蓋了第一個`func`,但是第二個script標簽中的`func`并沒有覆蓋上面的第二個`func`。所以說預解析是分段的。
tip:但是要注意,分段只是單純的針對函數(shù),變量并不會分段預解析。
這里會直接報錯,`func is not a function`,原因就是函數(shù)表達式,并不會被提升。只是簡單地當做變量聲明進行了處理,如下:
上面這段代碼,就是所謂的條件式函數(shù)聲明,這段代碼在Gecko引擎中打印`"undefined"`、`"function"`;而在其他瀏覽器中則打印`"function"`、`"function"`。
原因在于Gecko加入了ECMAScript以外的一個feature:條件式函數(shù)聲明。
Conditionally created functions Functions can be conditionally declared, that is, a function declaration can be nested within an if statement.
Note: Although this kind of function looks like a function declaration, it is actually an expression (or statement), since it is nested within another statement. See differences between function declarations and function expressions.
Note中的文字說明,條件式函數(shù)聲明的處理和函數(shù)表達式的處理方式一樣,所以條件式函數(shù)聲明沒有聲明提升的特性。
猜你喜歡: