深入理解Javascript作用域与变量提升


Posted in Javascript onDecember 09, 2013

下面的程序是什么结果?

var foo = 1;
function bar() {
 if (!foo) {
  var foo = 10;
 }
 alert(foo);
}
bar();

结果是10;

那么下面这个呢?

var a = 1;
function b() {
 a = 10;
 return;
 function a() {}
}
b();
alert(a);

结果是1.

吓你一跳吧?发生了什么事情?这可能是陌生的,危险的,迷惑的,同样事实上也是非常有用和印象深刻的javascript语言特性。对于这种表现行为,我不知道有没有一个标准的称呼,但是我喜欢这个术语:“Hoisting (变量提升)”。这篇文章将对这种机制做一个抛砖引玉式的讲解,但是,首先让我们对javascript的作用域有一些必要的理解。

Javascript的作用域

对于Javascript初学者来说,一个最迷惑的地方就是作用域;事实上,不光是初学者。我就见过一些有经验的javascript程序员,但他们对scope理解不深。javascript作用域之所以迷惑,是因为它程序语法本身长的像C家族的语言,像下面的C程序:

#include <stdio.h>
int main() {
 int x = 1;
 printf("%d, ", x); // 1
 if (1) {
  int x = 2;
  printf("%d, ", x); // 2
 }
 printf("%d\n", x); // 1
}

输出结果是1 2 1,这是因为C家族的语言有块作用域,当程序控制走进一个块,比如if块,只作用于该块的变量可以被声明,而不会影响块外面的作用域。但是在Javascript里面,这样不行。看看下面的代码:
var x = 1;
console.log(x); // 1
if (true) {
 var x = 2;
 console.log(x); // 2
}
console.log(x); // 2

结果会是1 2 2。因为javascript是函数作用域。这是和c家族语言最大的不同。该程序里面的if并不会创建新的作用域。

对于很多C,c++,java程序员来说,这不是他们期望和欢迎的。幸运的是,基于javascript函数的灵活性,这里有可变通的地方。如果你必须创建临时的作用域,可以像下面这样:

function foo() {
 var x = 1;
 if (x) {
  (function () {
   var x = 2;
   // some other code
  }());
 }
 // x is still 1.
}

这种方法很灵活,可以用在任何你想创建临时的作用域的地方。不光是块内。但是,我强烈推荐你花点时间理解javascript的作用域。它很有用,是我最喜欢的javascript特性之一。如果你理解了作用域,那么变量提升就对你显得更有意义。

变量声明,命名,和提升

在javascript,变量有4种基本方式进入作用域:

•1 语言内置:所有的作用域里都有this和arguments;(译者注:经过测试arguments在全局作用域是不可见的)

•2 形式参数:函数的形式参数会作为函数体作用域的一部分;

•3 函数声明:像这种形式:function foo(){};

•4 变量声明:像这样:var foo;

函数声明和变量声明总是会被解释器悄悄地被“提升”到方法体的最顶部。这个意思是,像下面的代码:

function foo() {
 bar();
 var x = 1;
}

实际上会被解释成:
function foo() {
 var x;
 bar();
 x = 1;
}

无论定义该变量的块是否能被执行。下面的两个函数实际上是一回事:
function foo() {
 if (false) {
  var x = 1;
 }
 return;
 var y = 1;
}
function foo() {
 var x, y;
 if (false) {
  x = 1;
 }
 return;
 y = 1;
}

请注意,变量赋值并没有被提升,只是声明被提升了。但是,函数的声明有点不一样,函数体也会一同被提升。但是请注意,函数的声明有两种方式:
function test() {
 foo(); // TypeError "foo is not a function"
 bar(); // "this will run!"
 var foo = function () { // 变量指向函数表达式
  alert("this won't run!");
 }
 function bar() { // 函数声明 函数名为bar
  alert("this will run!");
 }
}
test();

这个例子里面,只有函数式的声明才会连同函数体一起被提升。foo的声明会被提升,但是它指向的函数体只会在执行的时候才被赋值。

上面的东西涵盖了提升的一些基本知识,它们看起来也没有那么迷惑。但是,在一些特殊场景,还是有一定的复杂度的。

变量解析顺序

最需要牢记在心的是变量解析顺序。记得我前面给出的命名进入作用域的4种方式吗?变量解析的顺序就是我列出来的顺序。

<script>
function a(){ 
}
var a;
alert(a);//打印出a的函数体
</script>
<script>
var a;
function a(){ 
}
alert(a);//打印出a的函数体
</script>
//但是要注意区分和下面两个写法的区别:
<script>
var a=1;
function a(){ 
}
alert(a);//打印出1
</script>
<script>
function a(){ 
}
var a=1;
alert(a);//打印出1
</script>

这里有3个例外:

1 内置的名称arguments表现得很奇怪,他看起来应该是声明在函数形式参数之后,但是却在函数声明之前。这是说,如果形参里面有arguments,它会比内置的那个有优先级。这是很不好的特性,所以要杜绝在形参里面使用arguments;

2 在任何地方定义this变量都会出语法错误,这是个好特性;

3 如果多个形式参数拥有相同的名称,最后的那个具有优先级,即便实际运行的时候它的值是undefined;

命名函数

你可以给一个函数一个名字。如果这样的话,它就不是一个函数声明,同时,函数体定义里面的指定的函数名( 如果有的话,如下面的spam, 译者注)将不会被提升, 而是被忽略。这里一些代码帮助你理解:

foo(); // TypeError "foo is not a function"
bar(); // valid
baz(); // TypeError "baz is not a function"
spam(); // ReferenceError "spam is not defined"
var foo = function () {}; // foo指向匿名函数
function bar() {}; // 函数声明
var baz = function spam() {}; // 命名函数,只有baz被提升,spam不会被提升。
foo(); // valid
bar(); // valid
baz(); // valid
spam(); // ReferenceError "spam is not defined"

怎么写代码

现在你理解了作用域和变量提升,那么这对于javascript编码意味着什么?最重要的一点是,总是用var定义你的变量。而且我强烈推荐,对于一个名称,在一个作用域里面永远只有一次var声明。如果你这么做,你就不会遇到作用域和变量提升问题。

语言规范怎么说

我发现ECMAScript参考文档总是很有用。下面是我找到的关于作用域和变量提升的部分:

如果变量在函数体类声明,则它是函数作用域。否则,它是全局作用域(作为global的属性)。变量将会在执行进入作用域的时候被创建。块不会定义新的作用域,只有函数声明和程序(译者以为,就是全局性质的代码执行)才会创造新的作用域。变量在创建的时候会被初始化为undefined。如果变量声明语句里面带有赋值操作,则赋值操作只有被执行到的时候才会发生,而不是创建的时候。

我期待这篇文章会对那些对javascript比较迷惑的程序员带来一丝光明。我自己也尽最大的可能去避免带来更多的迷惑。如果我说错了什么,或者忽略了什么,请告知。

译者补充

有位朋友提醒了我发现了IE下全局作用域下命名函数的提升问题:

我翻译文章的时候是这么测试的:

<script>
functiont(){
spam();
var baz = function spam() {alert('this is spam')};
}
t();
</script>

这种写法, 即非全局作用域下的命名函数的提升,在ie和ff下表现是一致的. 我改成:
<script>
spam();
var baz = function spam() {alert('this is spam')};
</script>

则ie下是可以执行spam的,ff下不可以. 说明不同浏览器在处理这个细节上是有差别的.

这个问题还引导我思考了另2个问题,1:对于全局作用于范围的变量,var与不var是有区别的. 没有var的写法,其变量不会被提升。比如下面两个程序,第二个会报错:

<script>
alert(a);
var a=1;
</script>

<script>
alert(a);
a=1;
</script>

2: eval中创建的局部变量是不会被提升的(它也没办法做到).
<script>
var a = 1;
function t(){
 alert(a);
 eval('var a = 2');
 alert(a);
}
t();
alert(a);
</script>
Javascript 相关文章推荐
从零开始学习jQuery (三) 管理jQuery包装集
Feb 23 Javascript
js点击出现悬浮窗效果不使用JQuery插件
Jan 20 Javascript
两种方法基于jQuery实现IE浏览器兼容placeholder效果
Oct 14 Javascript
jQuery验证插件validation使用指南
Apr 21 Javascript
JavaScript中清空数组的方法总结
Dec 02 Javascript
jQuery实现立体式数字动态增加(animate方法)
Dec 21 Javascript
浅谈js算法和流程控制
Dec 29 Javascript
Vue.js项目部署到服务器的详细步骤
Jul 17 Javascript
纯html+css+javascript实现楼层跳跃式的页面布局(实例代码)
Oct 25 Javascript
在vue-cli中组件通信的方法
Dec 16 Javascript
Vue 监听列表item渲染事件方法
Sep 06 Javascript
vue+element实现图片上传及裁剪功能
Jun 29 Javascript
Javascript全局变量var与不var的区别深入解析
Dec 09 #Javascript
jquery div拖动效果示例代码
Dec 08 #Javascript
jquery垂直公告滚动实现代码
Dec 08 #Javascript
jquery中交替点击事件toggle方法的使用示例
Dec 08 #Javascript
JavaScript 判断用户输入的邮箱及手机格式是否正确
Dec 08 #Javascript
jqplot通过ajax动态画折线图的方法及思路
Dec 08 #Javascript
JavaScript 32位整型无符号操作示例
Dec 08 #Javascript
You might like
PHP的范围解析操作符(::)的含义分析说明
2011/07/03 PHP
利用php递归实现无限分类 格式化数组的详解
2013/06/08 PHP
关于php支持分块与断点续传文件下载功能代码
2014/05/09 PHP
CI框架(ajax分页,全选,反选,不选,批量删除)完整代码详解
2016/11/01 PHP
PHP实现字符串翻转功能的方法【递归与循环算法】
2017/11/03 PHP
ie 处理 gif动画 的onload 事件的一个 bug
2007/04/12 Javascript
javascript 自动转到命名锚记
2009/01/10 Javascript
Jquery 常用方法经典总结
2010/01/28 Javascript
JavaScript 面向对象的之私有成员和公开成员
2010/05/04 Javascript
基于jQuery实现图片的前进与后退功能
2013/04/24 Javascript
Jquery多选框互相内容交换的实例代码
2013/07/04 Javascript
jQuery实现鼠标经过图片预览大图效果
2014/04/10 Javascript
JS实现一个列表中包含上移下移删除等功能
2014/09/24 Javascript
JavaScript中使用typeof运算符需要注意的几个坑
2014/11/08 Javascript
全面解析bootstrap格子布局
2016/05/22 Javascript
AngularJS 执行流程详细介绍
2016/08/18 Javascript
解决vue2.0 element-ui中el-upload的before-upload方法返回false时submit()不生效问题
2018/08/24 Javascript
三种Webpack打包方式(小结)
2018/09/19 Javascript
layui 数据表格复选框实现单选功能的例子
2019/09/19 Javascript
js 下拉菜单点击旁边收起实现(踩坑记)
2019/09/29 Javascript
对vuex中store和$store的区别说明
2020/07/24 Javascript
NestJs使用Mongoose对MongoDB操作的方法
2021/02/22 Javascript
python 获取et和excel的版本号
2009/04/09 Python
Python使用openpyxl读写excel文件的方法
2017/06/30 Python
Pytorch之保存读取模型实例
2019/12/30 Python
python实现图片横向和纵向拼接
2020/03/05 Python
网络艺术零售业的先驱者:artrepublic
2017/09/26 全球购物
adidas澳大利亚官方网站:adidas Australia
2018/04/15 全球购物
机械专业个人求职自荐信格式
2013/09/21 职场文书
大学生活动策划方案
2014/02/10 职场文书
《三峡》教学反思
2014/03/01 职场文书
餐饮投资计划书
2014/04/25 职场文书
2014年小学安全工作总结
2014/12/04 职场文书
《水上飞机》教学反思
2016/02/20 职场文书
互联网的下一个风口:新的独角兽将诞生
2019/08/02 职场文书
HTML5之高度塌陷问题的解决
2022/06/01 HTML / CSS