让let语句在全局对象上创建属性吗?

Ben*_*aum 41 javascript let ecmascript-6

在JavaScript中,var声明在全局对象上创建属性:

var x = 15;
console.log(window.x); // logs 15 in browser
console.log(global.x); // logs 15 in Node.js
Run Code Online (Sandbox Code Playgroud)

ES6引入let了具有块范围的声明的词法作用域.

let x = 15;
{
   let x = 14;
}
console.log(x); // logs 15;
Run Code Online (Sandbox Code Playgroud)

但是,这些声明是否在全局对象上创建属性?

let x = 15;
// what is this supposed to log in the browser according to ES6?
console.log(window.x); // 15 in Firefox
console.log(global.x); // undefined in Node.js with flag
Run Code Online (Sandbox Code Playgroud)

Fel*_*ing 39

Do let语句是否在全局对象上创建属性?

根据规范,没有:

全局环境记录在逻辑上是单个记录,但它被指定为封装对象环境记录声明性环境记录的组合.该对象环境记录作为其基本对象相关的全局对象境界.此全局对象是全局环境记录的GetThisBinding具体方法返回的值.全局环境记录的对象环境记录组件包含所有内置全局变量(第18节)的绑定以及全局代码中包含的FunctionDeclaration,GeneratorDeclarationVariableStatement引入的所有绑定.全局代码中所有其他ECMAScript声明的绑定包含在全局环境记录的声明性环境记录组件中.

更多解释:

  • 声明环境记录存储在内部数据结构中的绑定.不可能以任何方式掌握该数据结构(考虑功能范围).

  • 一个对象环境记录使用的实际JS对象作为数据结构.对象的每个属性都成为绑定,反之亦然.全局环境具有对象环境对象,其"绑定对象"是全局对象.另一个例子是with.

现在,正如引用部分所述,只有FunctionDeclaration,GeneratorDeclarationVariableStatement在全局环境的对象环境记录中创建绑定.即只有这个绑定成为全局对象的属性.

所有其他声明(例如constlet)都存储在全局环境的声明性环境记录中,该记录不基于全局对象.

  • 澄清一些事情:声明性记录和对象记录之间的最大区别在于声明性记录使用内部数据结构来存储绑定.对象记录使用实际的JS对象作为"存储".在ES5中,全局环境被定义为对象记录(http://es5.github.io/#x10.2.3),这就是为什么全局范围内的所有变量声明都成为全局对象的属性的原因.但是在ES6中,全局环境由两种不同的环境记录类型组成,只有`var`声明存储在对象记录中,`let`声明不存储. (5认同)

Fli*_*imm 6

无论letvar变量,如果在脚本的顶层声明,是该脚本文件的访问之外。但是,只有var变量会分配给window对象。看看这个代码片段作为证明:

<script>
  var namedWithVar = "with var";
  let namedWithLet = "with let";
</script>

<script>
  console.log("Accessed directly:");
  console.log(namedWithVar);        // prints: with var
  console.log(namedWithLet);        // prints: with let
  console.log("");

  console.log("Accessed through window:");
  console.log(window.namedWithVar); // prints: with var
  console.log(window.namedWithLet); // prints: undefined
</script>
Run Code Online (Sandbox Code Playgroud)


Tra*_*er1 5

根据规范

“let 和 const 声明定义了作用域为运行执行上下文的 LexicalEnvironment 的变量。”

这意味着您应该能够访问执行范围内的变量,但不能访问外部。这将执行范围扩展到了经典的 JS 闭包结构(function-only 或 global)之外。

let全局定义一个变量可以解释这一点,正如您在 Firefox 中看到的那样,它绑定了一个全局变量,而 V8/iojs 则没有。

值得一提的是,在iojs中console.log(typeof x)number返回。在实践中,你不应该在模块之外定义变量,或者尽可能多的函数......尤其是constlet

  • *“在全局范围内定义一个 let 变量使这有待解释”* 正如您从我的回答中看到的那样,事实并非如此。你说的是对的,我们确实有更多的东西可以创建范围(还有模块),但是,仅此而已并不能回答提出的问题。 (3认同)