js之作用域链

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

  • 聊到js的作用域,就不得不聊到js中的作用域链。作用域链的用途是保证对执行环境有权访问的所有变量和函数的有序访问,本质是一个指向变量对象的指针列表,那是什么又是变量对象?
  • 这里要引入一个极为重要的概念:执行环境(execution context)也称执行上下文,执行环境定义了变量或函数有权访问的其他数据,每个执行环境都有一个与之关联的变量对象,环境中定义的所有变量和函数都保存在这个对象中。
  • js中有三种执行环境,只讨论两种,一种是全局执行环境,一种是函数执行环境,每个函数都有自己的执行环境,和一个关联的变量对象(局部环境中的变量对象只会在函数执行的过程中存在,而全局变量对象始终存在),创建函数时,会创建一个预先包含全局变量对象的作用域链,保存在函数的一个内部属性[[scope]]中,当某个函数第一次被调用时,会创建一个执行环境,通过复制函数的[[scope]]属性中的对象构建起执行环境的作用域链,然后又有一个活动对象(充当变量对象使用)被创建并被推进执行环境作用域链的前端。
  • 对于嵌套函数还有闭包而言,内部函数的作用域链就是这样是通过复制上一层函数的作用域链来构建的,然后,用this,arguments和其他命名参数的值来初始化函数的活动对象,然后把这个对象推至当前执行环境的作用域链的前端,这样这条函数作用域链就包含了自己本地的活动对象,上一层函数的变量对象,一直向上到全局环境的变量对象
  • 这也解释了闭包的原理,闭包是指有权访问另一个函数作用域中的变量的函数,在一个函数内部创建另一个函数是创建闭包的常见方式。函数内部的变量可以访问包含函数的变量。

(如果在当前作用域中没有查到值,就会向上级作用域去查,直到查到全局作用域,这么一个查找过程形成的链条就叫做作用域链)

总结:

  • 所谓的作用域链就是在函数的当前执行环境中中查找变量,如果查找不到,就会向上一级函数中的执行环境中查找,一直引申到全局执行环境,这样一个向上查找的过程就形成了作用链
  • 再展开一点来讲的话就是,每创建一个函数都会创建相应的执行环境,这个执行环境又有一个与之关联的变量对象,函数中定义的变量和内部函数,都是保存在这个变量对象里面的,然后作用域链就是一个指向变量对象的指针列表,这个列表里面,最开头指针指向的就是当前函数的变量对象,然后一直向后指向的是上一级函数的变量对象,然后再上一级别函数的变量对象,一直到全局执行环境的变量对象,如果在当前的变量对象中查找不到变量,就只能去查找上一 级函数的变量 对象。

你可能感兴趣的:(js之作用域链)