JavaScript 通过函数管理作用域。在函数内部声明的变量只在这个函数内部,函数外面不可用。另一方面,全局变量就是在任何函数外面声明的或是未声明直接简单使用的。
“未声明直接简单使用”,指的是不用 var
关键字来声明变量。这个我们已经非常清楚,避免造成隐式产生全局变量的方法就是声明变量尽量用 var
关键字。
可你以为用了 var
就 ok 了?来看看这个陷进:
function foo() {
var a = b = 0;
//
body…
}
也许你期望得到的是两个局部变量,但 b
却是货真价实的全局变量。why?
Because 赋值运算是自右往左的 ,所以这相当于:
function foo() {
var a = (b = 0);
//
body…
}
所以 b
是全局变量。
解决:变量声明,最好一个个来,别搞批发~_~;
变量声明
先来看陷阱:
myName = “global”;
function foo() {
alert(myName);
var myName = “local”;
alert(myName);
}
foo();
乍看上去,我们预计期望两次 alert
的结果分别为 “global” 与 “local”,但真实的结果是 “undefined” 与 “local”。why? Because 变量在同一作用域(同一函数)中,声明都是被提至作用域顶部先进行解析的。
所以以上代码片段的执行行为可能就像这样:
function foo() {
var myName;
alert(myName); //
“undefined”
myName = “local”;
alert(myName); //
“local”
}
用另一个陷阱来测试下你是否真的理解了预解析:
if (!(“a” in window)) {
var a = 1;
}
alert(a);
a
变量的声明被提前到了代码顶端,此时还未赋值。接下来进入 if
语句,判断条件中 "a"
已成立(
in windowa
已被声明为全局变量),所以判断语句计算结果为 false
,直接就跳出 if
语句了,所以 a
的值为 undefined
。
var a; //
“undefined”
console.log(“a” in window); //
true
if (!(“a” in window)) {
var a = 1; //
不执行
}
alert(a); //
“undefined”
解决:变量声明,最好手动置于作用域顶部,对于无法当下赋值的变量,可采取先声明后赋值的手法。
函数声明
函数声明也是被提前至作用域顶部,先于任何表达式和语句被解析和求值的
alert(typeof foo); //
“function”
function foo() {
//
body…
}
可以对比一下:
alert(typeof foo); //
“undefined”
var foo = function ()