1.作用域

作用域是根据名称找变量的一套规则。

变量的赋值操作会执行两个动作,首先编译器会在当前作用域中声明一个变量(如果之前没有声明过),然后在运行时引擎会在作用域中查找该变量,如果能够找到就会对它赋值。

引擎在查找变量时执行怎样的查找,会影响最终的查找结果。

当变量出现在赋值操作的左侧时进行LHS查询,出现在右侧时进行RHS查询:

  • console.log(a)对a的引用是一个RHS引用,这里a并没有赋予任何值。相应地,需要查找并取得a的值,这样才能将值传递给console.log(..)

  • a = 2对a的引用是LHS引用,因为实际上我们并不关心当前的值是什么,只是 为=2这个赋值操作找到一个目标。

当一个块或函数嵌套在另一块或函数中时,就发生了作用域的嵌套。在当前作用域中无法找到某个变量时,引擎就会在外层嵌套的作用域中继续查找,直到找到该变量,或抵达最外层的作用域为止。

2.函数作用域

2.1 函数中的作用域

function foo(a){
var b = 2;
function bar(){
//...
}
//更多代码
var c = 3;
}

foo(...)的作用域气泡中包含了标识符a、b、c和bar。

bar(...)拥有自己的作用域气泡。全局作用域也有自己的作用域气泡,它只包含了一个标识符foo。

由于a、b、c和bar都附属于foo(...)的作用域气泡,因此无法从foo(...)的外部对它们进行访问。

函数作用域的含义是指,属于这个函数的全部变量都可以在整个函数的范围内使用及复用。

2.2 隐藏内部实现

“隐藏”作用域中的变量和函数好处是可以避免标识符之间的冲突。

当程序中加载了多个第三方库时,如果它们没有妥善地将内部私有的函数或变量隐藏起来,就会很容易引发变量冲突。

这些库通常会在全局作用域中声明一个名字足够独特的变量,通常是一个对象。这个对象被用作库的命名空间,所有需要暴露给外界的功能都会成为这个对象的属性。

var MyReallyCoolLibrary = {
awesome:"stuff",
doSomething:function(){
//...
},
doAnotherThing:function(){
//...
}
};

2.3 函数作用域

函数声明函数表达式之间最重要的区别是它们的名称标识符将会绑定在何处。

var a = 2;
//函数声明
function foo(){
var a = 3;
console.log(a);//3
}
foo();
console.log(a);//2
var a = 2;
//函数表达式
(function foo(){
var a = 3;
console.log(a);//3
})();
console.log(a);//2

(function foo(){...})作为函数表达式意味着foo只能在...所代表的位置中被访问,外部作用域则不行。

匿名函数表达式:

setTimeout(function(){
console.log("I waited 1 second!");
},1000);

函数表达式可以是匿名的,而函数声明则不可以省略函数名。

匿名函数在栈追踪中不会显示出有意义的函数名,使得调试很困难。

始终给函数表达式命名是一个最佳实践。

setTimeout(function() timeoutHandler{
console.log("I waited 1 second!");
},1000);

IIFE(Immediately-invoked function expression),即立即执行函数表达式

var a = 2;
(function foo(){
var a = 3;
console.log(a);//3
})();
console.log(a);//2

由于函数被包含在一对()括号内部,因此成为了一个表达式,通过在末尾加上另外一个()可以立即执行这个函数。

IIFE还有一种变化的用途是倒置代码的运行顺序,将需要运行的函数放在第二位,在IIFE执行之后当作参数传递进去。

var a = 2;
(function IIFE(def){
def(window);
})(function def(global){
var a = 3;
console.log(a);//3
console.log(global.a);//2
});

2.4 提升

引擎会在解释JavaScript代码之前先对其进行编译,编译阶段中的一部分工作就是找到所有的声明,并用合适的作用域将它们关联起来。

包括变量和函数在内的所有声明都会在任何代码被执行前首先被处理。

a = 2;
var a;
console.log(a);//输出结果为2 /*
等价于
var a;
a = 2;
console.log(a);//输出结果为2
*/
console.log(a);//输出结果为undefined
var a = 2; /*
等价于
var a;
console.log(a);//输出结果为undefined
a = 2;
*/

上面的处理过程好像变量和函数声明从它们在代码中出现的位置被“移动”到了最上面,这个过程就叫作提升

每个作用域都会进行提升操作。

函数声明和变量都会被提升。但是一个值得注意的细节是函数会首先被提升,然后才是变量

foo();//1
var foo;
function foo(){
console.log(1);
}
foo = function(){
console.log(2);
}; /*
等价于
function foo(){
console.log(1);
}
foo();//1
foo = function(){
console.log(2);
};
*/

注意:var foo尽管出现在function foo()...的声明之前,但它是重复的声明(因此被忽略了),因为函数声明会被提升到普通变量之前。

尽管重复的var声明会被忽略掉,但出现在后面的函数声明还是可以覆盖前面的。

foo();//3
function foo(){
console.log(1);
}
var foo = function(){
console.log(2);
};
function foo(){
console.log(3);
}

在同一个作用域中进行重复定义是非常糟糕的,而且经常会导致各种奇怪的问题。

3.作用域闭包

JavaScript中闭包无处不在,你只需要能够识别并拥抱它。

闭包是基于词法作用域书写代码时所产生的自然结果,你甚至不需要为了利用它们而有意识地创建闭包。

function foo(){
var a = 2;
function bar(){
console.log(a);
}
return bar;
}
var baz = foo();
baz();//2--这就是闭包的效果

函数bar()的词法作用域能够访问foo()的内部作用域,在foo()执行后,其返回值(也就是内部的bar()函数)赋值给变量baz并调用baz()

foo()执行后,通常会期待foo()的整个内部作用域都被销毁,因为我们知道引擎有垃圾回收器用来释放不再使用的内存空间。由于看上去foo()的内容不会再被使用,所以很自然地会考虑对其进行回收。

bar()拥有涵盖foo()内部作用域的闭包,使得该作用域能够一直存活,以供bar()在之后任何时间进行引用。

bar()依然持有对该作用域的引用,而这个引用就叫作闭包

function foo(){
var a = 2;
function baz(){
console.log(a);
}
bar(baz);
}
function bar(fn){
fn();//这就是闭包!
}
var fn;
function foo(){
var a = 2;
function baz(){
console.log(a);
}
fn = baz;//将baz分配给全局变量
}
function bar(){
fn();//这就是闭包!
}
foo();
bar();//2

JS的作用域和闭包的更多相关文章

  1. JS之作用域与闭包

    JS之作用域与闭包   作用域在JS中同样也是一个重要的概念.它不复杂,因为ES5中只有全局作用域和函数作用域,我们都知道他没有块级作用域.但在ES6中多了一个let,他可以保证外层块不受内层块的影响 ...

  2. 解析js中作用域、闭包——从一道经典的面试题开始

    如何理解js中的作用域,闭包,私有变量,this对象概念呢? 就从一道经典的面试题开始吧! 题目:创建10个<a>标签,点击时候弹出相应的序号 先思考一下,再打开看看 //先思考一下你会怎 ...

  3. 你不知道的JS之作用域和闭包 附录

     原文:你不知道的js系列 A 动态作用域 动态作用域 是和 JavaScript中的词法作用域 对立的概念. 动态作用域和 JavaScript 中的另外一个机制 (this)很相似. 词法作用域是 ...

  4. 你不知道的JS之作用域和闭包(五)作用域闭包

    原文:你不知道的js系列 一个简单粗暴的定义 闭包就是即使一个函数在它所在的词法作用域外部被执行,这个函数依然可以访问这个作用域. 比如: function foo() { var a = 2; fu ...

  5. JS(作用域和闭包)

    1.对变量提升的理解 1.变量定义(上下文) 2.函数声明 2.说明 this 几种不同的使用场景 常见用法 1.作为构造函数执行 2.作为对象属性执行 3.作为普通函数执行(this === win ...

  6. 你不知道的JS之作用域和闭包(三)函数 vs. 块级作用域

      原文:你不知道的js系列 在第(二)节中提到的,标识符在作用域中声明,这些作用域就像是一个容器,一个嵌套一个,这个嵌套关系是在代码编写时定义的. 那么到底是什么产生了一个新的作用域,只有函数能做到 ...

  7. js中作用域和闭包

    作用域链实例   (1) function example() { var age = 23; alert(age) } var age = 25; example(); alert(age); // ...

  8. js变量作用域和闭包的示例

    <script> /* js是函数级作用域,在函数内部的变量,内部都能访问, 外部不能访问内部的,但是内部可以访问外部的变量 闭包就是拿到本不该属于他的东西,闭包会造成内存泄漏,你不知道什 ...

  9. 你不知道的JS之作用域和闭包(四)(声明)提升

    原文:你不知道的js系列 先有鸡还是先有蛋? 如下代码: a = 2; var a; console.log( a ); 很多开发者可能会认为结果会输出 undefined,因为 var a 在 a ...

随机推荐

  1. MYSQL数据库与Emoji表情的故事

    问题背景 手机上众多输入法和键盘支持输入 emoji 表情,给早期设计的程序造成了越来越多的干扰. 移动端购物的流行,2018 年 "双十一"全网移动端交易达到 93.6% 微信年 ...

  2. etcd集群证书安装过程一

    为确保安全,kubernetes 系统各组件需要使用 x509 证书对通信进行加密和认证. CA (Certificate Authority) 是自签名的根证书,用来签名后续创建的其它证书. 本文档 ...

  3. 微软职位内部推荐-SW Engineer II for Enterprise Platform

    微软近期Open的职位: Job posting title: SDE II Location: China, Beijing Group Overview Discovery & Colla ...

  4. API验证

    API验证说明 API验证: a. 发令牌: 静态 PS: 隐患 key被别人获取 b. 动态令牌 PS: (问题越严重)用户生成的每个令牌被黑客获取到,都会破解 c. 高级版本 PS: 黑客网速快, ...

  5. servlet基础学习总结

    Servlet的任务 1.  读取客户端发送的显示的数据,包括HTML表单和一些客户端程序的表单 2.  读取客户端发送的隐式的数据,包括cookies.媒体类型等 3.  处理数据并产生结果 4.  ...

  6. 调研ANDRIOD平台的开发环境的发展演变

    在同学的推荐下,我选用学习eclipse这个软件,参考了这个网址的教程开始了一步一步的搭建之路. http://jingyan.baidu.com/article/bea41d437a41b6b4c5 ...

  7. unique STL讲解和模板

    unique()是C++标准库函数里面的函数,其功能是去除相邻的重复元素(只保留一个),所以使用前需要对数组进行排序. 代码: #include<bits/stdc++.h> using ...

  8. Java join & yield

    Thread.yield()方法作用是:暂停当前正在执行的线程对象,并执行其他线程. yield()应该做的是让当前运行线程回到可运行状态,以允许具有相同优先级的其他线程获得运行机会.因此,使用yie ...

  9. 最新版ABP 动态WebAPI 日期转json带T的解决方案| ABP DateTIme Json format

    ABP动态webapi返回的json数据中,日期时间带T还有毫秒数的问题,在以往的版本中可以使用下面方法解决: 在XXXAbpWebApiModule中加上下面的代码: 很老的很老的版本有效: pub ...

  10. JS选取DOM元素的方法

    摘自JavaScript权威指南(jQuery根据样式选择器查找元素的终极方式是 先用getElementsByTagName(*)获取所有DOM元素,然后根据样式选择器对所有DOM元素进行筛选) 今 ...