下面我有兩行甜美而簡單的程式碼。但我可以向你保證,它要么會讓你很困惑(因為你忽略了 JS 的下劃線原則),要么安慰你。
但它已經載入瞭如下知識概念
我的矛盾聲明
就像 var、const 和 let 也提升了它們的屬性,但它們位於不同的區域。
吊掛Def(簡單/外行版)
現在是時候深入了解 Js 如何編譯和執行我們的兩行程式碼了
在 JavaScript 中,編譯器和引擎處理變數宣告和賦值的方式可能有細微差別,尤其是在處理 let 和 var 時。
讓我們從編譯器和執行的角度來分解給定程式碼的過程:
name = 'ashu'; let name;
此時我明確表示,當我們寫 javascript 程式碼時,第一個解析器和編譯器會編譯我們的程式碼,然後進入執行階段。
編譯器視角
第一行: name = 'ashu';
編譯階段,
JavaScript 引擎解析程式碼並建立必要的範圍。
作業名稱 = 'ashu';
會注意到,但是在這個階段,引擎不會執行程式碼;它只是記錄對名為 name 的變數的賦值的存在。
如果之前沒有宣告過 name,編譯器會將其視為對全域變數(全域作用域中的 var name)的賦值,因為 var 宣告是提升並全域可存取的。
第二行: 說出名字;
當編譯器遇到let名稱時;聲明,它承認 name 應該是區塊範圍的。
編譯器將名稱放入臨時死區(TDZ) 它所屬的範圍,
意思是承認名稱的存在,但將其標記為未初始化。
let 聲明與 var 的提升方式不同。
相反,它在作用域中建立一個綁定,並僅在執行宣告時初始化它。
執行視角
第一行: name = 'ashu';
當 JavaScript 引擎執行賦值 name = 'ashu'; 時,
它檢查目前範圍內是否存在名稱。由於 name 是用 let 宣告的,但位於 TDZ(暫時死區)中,在 let 宣告初始化之前存取它的任何嘗試都會導致 ReferenceError。
因此,此時name在TDZ中,賦值name = 'ashu';導致 ReferenceError.
第二行: 說出名字;
此行初始化區塊作用域內的 name 變數。
此後,名稱不再位於 TDZ 中,並且可以毫無錯誤地存取或分配。
現在獎勵小費
未宣告與未定義與未初始化之間的差異;
undeclare :- 變數尚未宣告。
未定義 :- 宣告但未初始化的變數;
未初始化:- 變數已定義,但其值將在稍後部分出現。
例如:- const result =multiplyBy2(5);
直到函數的回傳值將會被指派給結果,直到此時它將處於未初始化區域。
有趣的事實:-
你知道臨時死區最初是為 Const 裝飾的,但他們在 **Let**
中採用了較晚的時間點參考:-
以上是揭秘 JavaScript:深入探討提升、臨時死區與可變狀態的詳細內容。更多資訊請關注PHP中文網其他相關文章!