请参阅下面的示例代码
<script> alert(a); // undefined alert(b); // It is Error,b is not defined. var a=1; b=10; </script>
当变量a和b都在全局范围内时,为什么我收到b的错误消息.但变量a没有错误信息?是什么原因 ?
有人可以解释一下吗?
解决方法
第一个警报显示未定义,因为var语句被提升到封闭范围的顶部,换句话说,var语句和函数声明是在执行实际代码之前,在解析阶段.
执行代码时,相当于:
var a; // declared and initialized with `undefined` before the code executes alert(a); // undefined alert(b); // ReferenceError,b is not declared. a=1; b=10;
第二个警报甚至没有执行,尝试访问b会给你一个ReferenceError,因为你从未声明它,并且你正试图访问它.
这就是标识符解析过程在Javascript中的工作方式,如果在所有作用域链中找不到标识符,则抛出ReferenceError异常.
此外,您应该知道,在没有首先声明标识符的情况下分配标识符(如b = 10)在技术上并不声明变量,即使在全局范围内,效果也可能类似(并且似乎有效),最后标识符作为全局对象的属性结束,例如:
var a = 1; b = 10; // Similar effect: window.a; // 1 window.b; // 10
但这只是因为全局对象是范围链的最顶层环境记录.
上面两者之间的另一个区别是用var声明的标识符在全局对象上产生一个不可配置的属性(不能删除),例如:
delete window.a; // false delete window.b; // true
此外,如果您在函数的范围内,并且对未声明的标识符进行赋值,它将最终成为全局对象的属性,就像在上面的示例中一样,而var语句将创建一个局部变量,例如:
(function(){ var a = 1; b = 10; })(); typeof window.a; // 'undefined',was locally scoped in the above function typeof window.b; // 'number',leaked,an unintentional global
我真的不鼓励对未声明的标识符进行分配,总是使用var来声明你的变量,而且,在ECMAScript 5严格模式下不允许这样做,对未声明的标识符进行的赋值抛出一个ReferenceError:
(function(){'use strict'; b = 10;})(); // throws a ReferenceError