【12月5日】关于JS中的作用域和作用域链

作用域

定义:作用域是指程序源代码中定义变量的区域。
作用:作用域规定了如何查找变量,也就是确定当前执行代码对变量的访问权限。
在javaScript中的应用 :JavaScript采用词法作用域(lexical scoping),也就是静态作用域。
那什么又是 词法作用域或者静态作用域呢?

请继续往下看

静态作用域与动态作用域

因为javaScript采用的是词法作用域(也就是静态作用域),函数的作用域在函数定义的时候就决定了。

而词法作用域相对的是动态作用域,函数的作用域是在函数调用的时候才决定的。
让我们看一个例子来理解词法作用域和动态作用域之间的区别:

var value = 1;

function foo() {
  console.log(value);
}

function bar() {
  var value = 2;
  foo();
}

bar();

// 结果是 ???
上面的代码中:

1.我们首先定义了一个value,并赋值为1;
2.声明一个函数foo,函数的功能是打印 value 这个变量的值;
3.声明一个函数bar,函数内部重新创建了一个变量 value 这个变量赋值为2;
在函数内部执行了 foo() 这个函数;
4.执行 bar() 这个函数
假设javaScript采用静态作用域,让我们分析下执行过程:

执行foo函数,首先从 foo 函数内部查找是否有变量 value ,如果没有
就根据书写的位置,查找上面一层的代码,我们发现value等于1,所以结果会打印 1。

假设javaScript采用动态作用域,让我们分析下执行过程:

执行foo函数,依然是从 foo 函数内部查找是否有局部变量 value。如果没有,
就从调用函数的作用域,也就是 bar 函数内部查找 value 变量,所以结果会打印 2。

上面在区分静态作用于和动态作用域的时候,我们已经说了如果是静态作用域,那么函数在书写定义的时候已经确定了,而动态作用域是函数执行过程中才确定的。

JavaScript采用的是静态作用域,所以这个例子的结果是 1。
我们在控制台中输入执行上面的函数,检验一下执行结果果然是 1。

动态作用域

那什么语言是采用的动态的作用域呢? 其实bash 就是动态作用域,
我们可以新建一个 scope.bash 文件将下列代码放进去,执行一下这个脚本文件:

#!/bin/bash

value=1
function foo () {
    echo $value;
}
function bar () {
  local value=2;
  foo;
}
bar

上面代码运行的结果输出2很好解释,虽然在代码最上层定义了 value并赋值为1,但是在调用foo函数的时候,在查找 foo 内部没有 value 变量后,会在foo 函数执行的环境中继续查找,也就是在bar 函数中查找,很幸运我们找到了。~

看一道题

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

原因是:函数内的变量优先级高于全局变量,如果在函数内声明的局部变量和全局变量重名,那么在函数局部范围内,局部变量的赋值和计算将顶替全局变量原有的值,但是在函数之外继续引用全局变量,全局变量原来的值不变。
很容易理解
再直接打印一下scope,可以看到是没有改变的

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

这个时候把var去掉,全局scope被覆盖了

var scope = "global scope";
 function checkscope(){ 
    scope = "local scope"; //去掉了var声明
    return scope;
 } 
console.log(checkscope());
console.log(scope);//从global scope变成了local scope

再来看看当函数内嵌套多层的情况:

var scope = ' global scopr'; // 全局变量 现在值时global scope
function checkscope(){ // 第一层函数
     var scope = 'local scope'; //创建了一个函数内的全局变量
     function nested(){
         var scope = 'nested scope'; //创建了一个函数内的全局变量
         return scope; //这里放回 socope 的结果,是netsed scope 
     }
     return nested(); //返回netstd的值 也是 netsed scope

}
console.log(checkscope())// 所以这里的值是 nested scope 
var scope = "global scope";
function checkscope(){
  var scope = "local scope";
  function f(){
    return scope;
  }
  return f();
}
checkscope();

在checkscope 函数中 又定义一个函数 f ,这个函数 只做了一件事:返回scope 这个变量;
最后返回并执行 f 这个函数;
调用checkscope
按照javaScript中静态作用域理解,在执行 checkscope 这个函数的时候在函数内部执行的是f 这个函数,首先在 f 这个函数内部查找 scope 这个变量发现没有,继续在定义函数f的上面一层查找,发现在checkscope 这个函数作用域内 找到了scope的值 直接返回,至于 checkscope外面定义的scope没有理睬。

再举一个栗子

// 例2:
var scope = "global scope";
function checkscope(){
  var scope = "local scope";
  function f(){
    return scope;
  }
  return f;
}
checkscope()();

单纯的返回 f 这个函数;
调用checkscope
按照我们上面解释的javaScript中静态作用域理解,在执行 checkscope 这个函数的时候在函数内返回了函数f实际是在最外面调用的f但是由于javaScript是采用的词法作用域,因此函数的作用域基于函数创建的位置。

而引用《JavaScript权威指南》的回答就是:

JavaScript 函数的执行用到了作用域链,这个作用域链是在函数定义的时候创建的。嵌套的函数 f() 定义在这个作用域链里,其中的变量 scope 一定是局部变量,不管何时何地执行函数 f(),这种绑定在执行 f() 时依然有效。

但是在这里真正想让大家思考的是:

虽然两段代码执行的结果一样,但是两段代码究竟有哪些不同呢?

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容