Warning: file_get_contents(/data/phpspider/zhask/data//catemap/9/javascript/401.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181

Warning: file_get_contents(/data/phpspider/zhask/data//catemap/1/firebase/6.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181
Javascript 让语句在全局对象上创建属性吗?_Javascript_Ecmascript 6_Let - Fatal编程技术网

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

Javascript 让语句在全局对象上创建属性吗?,javascript,ecmascript-6,let,Javascript,Ecmascript 6,Let,在JavaScript中,var声明在全局对象上创建属性: var x = 15; console.log(window.x); // logs 15 in browser console.log(global.x); // logs 15 in Node.js ES6使用具有块作用域的声明引入词法作用域 let x = 15; { let x = 14; } console.log(x); // logs 15; 但是,这些声明是否在全局对象上创建属性 let x = 15; //

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

var x = 15;
console.log(window.x); // logs 15 in browser
console.log(global.x); // logs 15 in Node.js
ES6使用具有块作用域的声明引入词法作用域

let x = 15;
{
   let x = 14;
}
console.log(x); // logs 15;
但是,这些声明是否在全局对象上创建属性

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
:

“让和const声明定义作用于正在运行的执行上下文的LexicalEnvironment的变量。” 这意味着您应该能够在执行范围内访问变量,但不能在执行范围外访问。这将执行范围扩展到经典的JS闭包结构(仅函数或全局)之外

全局定义一个
let
变量可以对其进行解释,正如您在Firefox中看到的,它绑定了一个全局变量,而V8/iojs没有绑定

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

是否让
语句在全局对象上创建属性

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
根据报告,没有:

全局环境记录在逻辑上是单个记录,但它被指定为封装和的复合记录。的全局对象作为其基础对象。此全局对象是全局环境记录的
GetThisBinding
concrete方法返回的值。全局环境记录的对象环境记录组件包含所有内置全局()的绑定,以及全局代码中包含的FunctionDeclaration、GeneratorDeclaration或VariableStatement引入的所有绑定全局代码中所有其他ECMAScript声明的绑定都包含在全局环境记录的组件中。

还有一些解释:

  • 声明性环境记录将绑定存储在内部数据结构中。以任何方式都不可能掌握该数据结构(想想函数范围)

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

现在,正如引用的部分所述,只有FunctionDeclarations、GeneratorDeclarations和VariableStatements在全局环境的对象环境记录中创建绑定。也就是说,只有这个绑定才成为全局对象的属性

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

通过let关键字声明的变量不会在全局对象(浏览器窗口)上创建可访问的属性。 实际上,Firefox修复了它的行为:
let v=42;'窗口中的v//false

允许您声明在其上使用的块、语句或表达式范围有限的变量。这与关键字不同,该关键字全局定义变量,或局部定义整个函数,而不考虑块范围

let x = 15;
{
   let x = 14;
}
console.log(x); // logs 15;
在程序和函数的顶层,与var不同,let不在全局对象上创建属性。例如:

   var x = 'global';
   let y = 'global';
   console.log(this.x); // "global"
   console.log(this.y); // undefined
var x = 1;

if (x === 1) {
  var x = 2;

  console.log(x);
  // output: 2
}

console.log(x);
// output: 2
let x = 1;

if (x === 1) {
  let x = 2;

  console.log(x);
  // output: 2
}

console.log(x);
// output: 1
var声明的变量的作用域是其当前执行上下文,该上下文是封闭函数,对于在任何函数之外声明的变量,它是全局的。如果重新声明JavaScript变量,它不会丢失其值。例如:

   var x = 'global';
   let y = 'global';
   console.log(this.x); // "global"
   console.log(this.y); // undefined
var x = 1;

if (x === 1) {
  var x = 2;

  console.log(x);
  // output: 2
}

console.log(x);
// output: 2
let x = 1;

if (x === 1) {
  let x = 2;

  console.log(x);
  // output: 2
}

console.log(x);
// output: 1
<>强>注释:>与C、C++和java不同,JavaScript在使用<强> var > /St>>/P>声明变量时,没有块级别范围。 正如我们前面提到的那样,let
允许您声明变量,这些变量的作用域仅限于使用它的块、语句或表达式。例如:

   var x = 'global';
   let y = 'global';
   console.log(this.x); // "global"
   console.log(this.y); // undefined
var x = 1;

if (x === 1) {
  var x = 2;

  console.log(x);
  // output: 2
}

console.log(x);
// output: 2
let x = 1;

if (x === 1) {
  let x = 2;

  console.log(x);
  // output: 2
}

console.log(x);
// output: 1

在这里,我建议您阅读关于
let
var
变量(如果在脚本的顶层声明)的内容,它们都可以在脚本文件之外访问。但是,只有
var
变量被分配给
窗口
对象。请查看此代码片段作为证明:


var namedWithVar=“with var”;
let namedWithLet=“with let”;
console.log(“直接访问:”);
console.log(namedWithVar);//打印:使用var
console.log(namedWithLet);//印刷品:用let
控制台日志(“”);
日志(“通过窗口访问”);
console.log(window.namedWithVar);//打印:使用var
console.log(window.namedWithLet);//打印:未定义

@jfriend00是,它在Firefox和io.js中的行为也不同,这两种环境我都可以测试它。在node.js的repl中-如果你这样做
var x=15
,它将把它的范围扩大到全局范围,你可以通过在下一行键入
global.x
来轻松验证。问题:是否有
LexicalEnvironment
和相关的
EnvironmentRecord
与全局范围关联?回答我自己的问题:是和是。8.1第4段。我认为这是Firefox的相关bug报告:但8.1.1.1()表明
let
var
创建相同类型的环境记录(即,它们都创建
声明性环境记录
),这表明
let
var
在全局对象上的位置可能对应。但是,我认为全局情况可能很特殊。
var
不创建记录
let
var
绑定“存储”在环境记录中。还有不同类型的环境记录:声明性(您链接到的内容)、obj