下面我有两行甜蜜而简单的代码。但我可以向你保证,它要么会让你很困惑(因为你忽略了 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中文网其他相关文章!