javascript系列--javascript深刻理解--做用域,做用域链,闭包的面试题解

1、概要

做用域和做用域链是js中很是重要的特性,关系到理解整个js体系,闭包是对做用域的延伸,其余语言也有闭包的特性。html

那什么是做用域?做用域指的是一个变量和函数的做用范围。es6

一、js中函数内声明的全部变量在函数体内始终是可见的;面试

二、在ES6中有全局做用域和局部做用域,可是没有没有块级做用域(catch只在其内部生效);segmentfault

三、局部变量的优先级高于全局变量。数组

2、做用域

咱们来举几个栗子:闭包

2.1变量提高

var scope="global";
function scopeTest(){
    console.log(scope);
    var scope="local"  
}
scopeTest(); //undefined

上面的代码输出是undefined,这是由于局部变量scope变量提高了,等效于下面函数

var scope="global";
function scopeTest(){
    var scope;
    console.log(scope);
    scope="local"  
}
scopeTest(); //undefined

注意,若是在局部做用域中忘记var,那么变量就被声明为全局变量。ui

var scope="global";
function scopeTest(){
    console.log(scope);
    scope="local"  
}
scopeTest(); //global
var scope="global";
function scopeTest(){
    scope="local" 
    console.log(scope);
}
scopeTest(); //local

2.2没有块级做用域

和咱们其余经常使用语言不一样的是,js中没有块级做用域code

var data = [];

for (var i = 0; i < 3; i++) {
  data[i] = function () {
    console.log(i);
  };
}

data[0]();    // 3
data[1]();    // 3
data[2]();    // 3

2.3做用域链

每一个函数都有本身的执行上下文环境,当代码在这个环境中执行时候,会建立变量对象的做用域链,htm

那什么是做用域链?做用域链式是一个对象列表。

做用域链的做用?他保证了变量对象的有序访问。

做用域链开始的地方:当前代码执行环境的变量对象,常被称之为“活跃对象”(AO),变量的查找会从第一个链的对象开始,若是对象中包含变量属性,那么就中止查找,若是没有就会继续向上级做用域查找,直到找到全局对象中,若是找不到就会报ReferenceError。

2.4闭包

function createClosure(){
    var name = "jack";
    return {
        setStr:function(){
            name = "rose";
        },
        getStr:function(){
            return name + ":hello";
        }
    }
}
var builder = new createClosure();
builder.setStr();
console.log(builder.getStr()); //rose:hello

上面在函数中反悔了两个闭包,这两个闭包都维持着对外部做用域的引用,所以无论在哪调用都是可以访问外部函数中的变量。在一个函数内部定义的函数,闭包中会将外部函数的自由对象添加到本身的做用域中,因此能够经过内部函数访问外部函数的属性,这就是js模拟私有变量的一种方式。

注意:因为闭包会额外的附带函数的做用域(内部匿名函数携带外部函数的做用域),所以,闭包会比其余函数多占用些内存空间,过分使用会致使内存占用增长。

3、闭包面试题解

因为做用域链机制的影响,闭包只能取得内部函数的最后一个值,这引发了一个反作用,若是内部函数在一个循环中,那么变量的值始终为最后一个值。

var data = [];

for (var i = 0; i < 3; i++) {
  data[i] = function () {
    console.log(i);
  };
}

data[0]();    // 3
data[1]();    // 3
data[2]();    // 3

若是想强制返回逾期结果,怎么整?

方法一:当即执行函数

for (var i = 0; i < 3; i++) {
    (function(num) {
        setTimeout(function() {
            console.log(num);
        }, 1000);
    })(i);
}
// 0
// 1
// 2

方法二:返回一个匿名函数赋值

var data = [];

for (var i = 0; i < 3; i++) {
  data[i] = (function (num) {
      return function(){
          console.log(num);
      }
  })(i);
}

data[0]();    // 0
data[1]();    // 1
data[2]();    // 2

不管上是当即执行函数仍是返回一个匿名函数赋值,原理上都是由于变量的按值传递,因此会将变量i的值赋值给实参num,在匿名函数的内部又建立了一个用于访问num的匿名函数,这样每个函数都有一个num的副本,互不影响。

方法三:使用es6的let

var data = [];

for (let i = 0; i < 3; i++) {
  data[i] = function () {
    console.log(i);
  };
}

data[0]();
data[1]();
data[2]();

解释一下原理:

var data = [];// 建立一个数组data;

// 进入第一次循环
{

let i = 0; // 注意:由于使用let使得for循环为块级做用域
           // 这次 let i = 0 在这个块级做用域中,而不是在全局环境中
data[0] = function() {
    console.log(i);
};

}

循环时,let声明了i,因此整个块是块级做用域,那么data[0]这个函数就成了一个闭包,这里用{}表述,只是但愿经过它来讲明let存在的时候,这个for循环块是块级做用域,而不是全局做用域。

上面的块级做用域,就像函数做用域同样,寒暑表执行完毕,其中的变量会被销毁,可是由于这个代码块中存在一个闭包,闭包的做用域链中引用着块级做用域,因此在闭包被调用以前,这个块级做用域内部的变量不会被销毁。

// 进入第二次循环
{

let i = 1; // 由于 let i = 1 和上面的 let i = 0     
           // 在不一样的做用域中,因此不会相互影响
data[1] = function(){
     console.log(i);
};

}
当执行data[1]()时,进入下面的执行环境。

{

let i = 1; 
 data[1] = function(){
      console.log(i);
 };

}

在上面这个执行环境中,它会首先寻找该执行环境中是否存在i,没有找到,就沿着做用域链继续向上找,在其所在的块级做用域执行环境中,找到i=1,因而输出1。



## 4、思考题

代码1:

var scope = "global scope";
function checkscope(){

var scope = "local scope";
function f(){
    return scope;
}
return f;

}

checkscope()(); //local scope

代码2:

var scope = "global scope";
function checkscope(){

var scope = "local scope";
function f(){
    return scope;
}
return f;

}

var foo = checkscope();
foo(); //local scope

## 4、参考

一、https://segmentfault.com/a/1190000000618597

二、https://www.cnblogs.com/zhuzhenwei918/p/6131345.html
相关文章
相关标签/搜索