3.5 KiB
id, title, challengeType, forumTopicId, dashedName
id | title | challengeType | forumTopicId | dashedName |
---|---|---|---|---|
587d7b87367417b2b2512b40 | 比较 var 和 let 关键字的作用域 | 1 | 301195 | compare-scopes-of-the-var-and-let-keywords |
--description--
如果你不熟悉 let
,请查看 这个挑战。
使用 var
关键字声明变量时,它是全局声明的,如果在函数内部声明则是局部声明的。
let
关键字的行为类似,但有一些额外的功能。 在代码块、语句或表达式中使用 let
关键字声明变量时,其作用域仅限于该代码块、语句或表达式。
例如:
var numArray = [];
for (var i = 0; i < 3; i++) {
numArray.push(i);
}
console.log(numArray);
console.log(i);
此处控制台将显示值 [0, 1, 2]
和 3
。
使用 var
关键字,i
是全局声明的。 所以当 i++
被执行时,它会更新全局变量。 此代码类似于以下内容:
var numArray = [];
var i;
for (i = 0; i < 3; i++) {
numArray.push(i);
}
console.log(numArray);
console.log(i);
此处控制台将显示值 [0, 1, 2]
和 3
。
如果你创建一个函数,将它存储起来,稍后在使用 i
变量的 for
循环中使用。这么做可能会出现问题。 这是因为存储的函数将始终引用更新后的全局 i
变量的值。
var printNumTwo;
for (var i = 0; i < 3; i++) {
if (i === 2) {
printNumTwo = function() {
return i;
};
}
}
console.log(printNumTwo());
此处控制台将显示值 3
。
可以看到,printNumTwo()
打印了 3 而不是 2。 这是因为赋值给 i
的值已经更新,printNumTwo()
返回全局的 i
,而不是在 for 循环中创建函数时 i
的值。 let
关键字就不会出现这种现象:
let printNumTwo;
for (let i = 0; i < 3; i++) {
if (i === 2) {
printNumTwo = function() {
return i;
};
}
}
console.log(printNumTwo());
console.log(i);
在这里控制台将显示值 2
和一个错误提示 i is not defined
。
i
未定义,因为它没有在全局范围内声明。 它只在 for
循环语句中被声明。 printNumTwo()
返回了正确的值,因为 let
关键字创建了三个具有唯一值(0、1 和 2)的不同 i
变量在循环语句中。
--instructions--
修改这段代码,使 if
语句中声明的 i
变量与在函数的第一行声明的 i
变量是彼此独立的。 请注意不要在你的代码的任何地方使用 var
关键字。
这个练习旨在表明使用 var
与 let
关键字声明变量时作用域之间的区别。 当编写类似这个练习中的函数的时候,通常来说最好使用不同的变量名,以避免混淆。
--hints--
代码中不应该出现 var
。
assert(!code.match(/var/g));
if
语句中声明的变量 i
应该等于字符串 block scope
。
assert(code.match(/(i\s*=\s*).*\s*.*\s*.*\1('|")block\s*scope\2/g));
checkScope()
应该返回字符串 function scope
。
assert(checkScope() === 'function scope');
--seed--
--seed-contents--
function checkScope() {
var i = 'function scope';
if (true) {
i = 'block scope';
console.log('Block scope i is: ', i);
}
console.log('Function scope i is: ', i);
return i;
}
--solutions--
function checkScope() {
let i = 'function scope';
if (true) {
let i = 'block scope';
console.log('Block scope i is: ', i);
}
console.log('Function scope i is: ', i);
return i;
}