资讯专栏INFORMATION COLUMN

作用域和作用域链

MageekChiu / 1093人阅读

摘要:这里一共说了三种作用域,其实可以说是两种一种是全局作用域,而是局部作用域函数作用域块级作用域,块级作用域概念又包括了函数作用域。

作用域:

变量所在的上下文,指的是变量在哪些地方可以访问

对于JavaScript来说有全局作用域但是没有块级作用域,在ES6中引入了关键字let可以生成块作用域.见以下代码:

var value = true
if (value) {
  var age = 18
  console.log(`我今年${age}岁了`)
}
console.log (`我也是${age}岁了哦`)

//输出  "我今年18岁了"   "我也是18岁了哦"
// 可见在if()中不存在块级作用域

在这里if()里的变量具有全局作用域,全局皆可使用

var value = true
if (value) {
  let age = 18
  console.log(`我今年${age}岁了`)
}
console.log (`我也是${age}岁了哦`)

/*输出
  "我今年18岁了"
  "error"
  "ReferenceError: age is not defined*/

在这里使用关键字let 使 if () 块里的变量age产生了块级作用域,使得它只在这个块里生效.

JS中有函数作用域,指的是作用域在函数内部。这里一共说了三种作用域,其实可以说是两种:一种是全局作用域,而是局部作用域(函数作用域、块级作用域),块级作用域概念又包括了函数作用域。

简要说下几个作用域声明方式:

全局作用域:在所有函数外部使用var语句声明变量或者在声明变量时忽略var则会隐式转化为全局变量

函数作用域: 需要在函数内部使用var 声明变量才行

块级作用域: 在变量名前添加let语句声明(ES6)

作用域链
var a = "你好,我是a";
function scopeChain(a) {
  var b =1;
  function inScope(a) {
     var c = "蚂蚁"
     console.log(`大象爱${c}`)
     console.log(`我是最内层的函数,这里也可以使用a: ${a}`)
  }
  console.log(`能使用a吗?${a}`)
  inScope(a)
}
scopeChain(a)

这里a是全局作用域下的变量,b是函数scopeChain()作用域下的变量,而c是函数scopeChain()里的inScope()函数作用域下的变量。

作用域链的前端始终是当前环境作用域下变量对象,逐层往外作用域链接,最后端是全局变量环境下的变量,这些变量时链接在一起,在解析一个变量时从链前端往后端搜索(从内不找外部找),但是有一点值得注意:每个变量的作用域总是从自身声明的作用域往外找,而不是调用它的地方

var a = 1
function fn1(){
  function fn3(){
    var a = 4
    fn2()
  }
  var a = 2
  return fn3
}
function fn2(){
  console.log(a)
}
var fn = fn1()
fn()

 //输出1

这里就是当fn1() 执行后调用 fn2() 时,发现fn3()作用域下没有,也就是作用域链前端没有,往外找一层也就是fn1()作用域下进行查找,作用域也就往后端前进了一步,发现还是没有,继续往外层作用域查找找到了全局作用域,也就是作用域链的最后端,找到了后调用它。

但是对于fn2()来说它需要调用a这个变量,这里也就出现了误区:在fn3()里有变量a,那么是用的是这个变量a吗?

但其实fn2()是发现不了这个变量a的,因为fn2()声明的地方并不在fn3()里,同理fn1()也不是fn2()声明的地方,所以对于fn2()来说它只发现了全局下的 var a =1 所以调用它并输出a时也就等于1.

var a = 1
function fn1(){
  function fn2(){
    console.log(a)
  }
  function fn3(){
    var a = 4
    fn2()
  }
  var a = 2
  return fn3
}
var fn = fn1()
fn()

//输出多少?

由以上的论述可以分析出这段代码,当fn1()被调用时也就是 return fn3 ,也就调用了fn3(),然后fn3()里又是调用fn2(),而fn() 是在fn1()里声明的,自然也就使用了fn1()里的变量a,又因为a变量在调用前声明并赋值了,故此输出为2

var a = 1
function fn1(){

  function fn3(){
    function fn2(){
      console.log(a)
    }
    var a

    fn2()
    a = 4
  }
  var a = 2
  return fn3
}
var fn = fn1()
fn()

//输出多少?

分析这段代码,发现与上面代码不同之处在于先声明了 ` var a 后没有里脊赋值,在调用了fn2()后再进行的赋值,那么这里应该是多少呢?

这里也就牵涉到了声明前置,对于fn3()下,当声明 var a时,也就是执行到了fn3()代码前,函数声明和变量声明会提前至代码前端,所以这里声明并没有影响到输出值得改变,但是赋值操作是按照程序顺序执行的,当调用前,a只声明没有赋值,则会输出undefined。 而具体变量查找是符合作用域链的顺序来的.

总结如下:

函数在执行的过程中,先从自己内部找变量

如果找不到,再从创建当前函数所在的作用域去找, 以此往上

注意找的是变量的当前的状态

个人学习备忘,如有谬误,欢迎指正。

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/94373.html

相关文章

  • 理解JavaScript中的作用域和作用域链

    摘要:示例当一个函数创建后,它的作用域链会被创建此函数的作用域中可访问的数据对象填充。每一个运行期上下文都和一个作用域链关联。此时,作用域链中函数的所有局部变量所在的作用域对象会被推后,访问代价变高了。 作用域 作用域就是变量与函数的可访问范围,即作用域控制着变量与函数的可见性和生命周期。在JavaScript中,变量的作用域有全局作用域和局部作用域两种。 作用域链 函数对象有一个内部属性[...

    XanaHopper 评论0 收藏0
  • JavaScript 作用域和作用域链学习

    摘要:作用域与作用域链每个函数都有自己的执行环境。这是初步了解作用域,如想更深入了解作用域,请看下面链接作用域原理作用域链由一道题图解的作用域或者看权威指南和高级程序设计 本文是我学习JavaScript作用域整理的笔记,如有不对,请多指出。 作用域 一个变量的作用域是程序源代码中定义这个变量的区域。 而在ES5中只分为全局作用域和函数作用域,也就是说for,if,while等语句是不会创建...

    史占广 评论0 收藏0
  • 深入理解JavaScript作用域和作用域链

    前言 JavaScript中有一个被称为作用域(Scope)的特性。虽然对于许多新手开发者来说,作用域的概念并不是很容易理解,本文我会尽我所能用最简单的方式来解释作用域和作用域链,希望大家有所收获! 想阅读更多优质文章请猛戳GitHub博客 作用域(Scope) 1.什么是作用域 作用域是在运行时代码中的某些特定部分中变量,函数和对象的可访问性。换句话说,作用域决定了代码区块中变量和其他资源的可见...

    baiy 评论0 收藏0
  • 深入理解JavaScript作用域和作用域链

    前言 JavaScript中有一个被称为作用域(Scope)的特性。虽然对于许多新手开发者来说,作用域的概念并不是很容易理解,本文我会尽我所能用最简单的方式来解释作用域和作用域链,希望大家有所收获! 想阅读更多优质文章请猛戳GitHub博客 作用域(Scope) 1.什么是作用域 作用域是在运行时代码中的某些特定部分中变量,函数和对象的可访问性。换句话说,作用域决定了代码区块中变量和其他资源的可见...

    ytwman 评论0 收藏0
  • 深入javascript——作用域和闭包

    摘要:注意由于闭包会额外的附带函数的作用域内部匿名函数携带外部函数的作用域,因此,闭包会比其它函数多占用些内存空间,过度的使用可能会导致内存占用的增加。 作用域和作用域链是javascript中非常重要的特性,对于他们的理解直接关系到对于整个javascript体系的理解,而闭包又是对作用域的延伸,也是在实际开发中经常使用的一个特性,实际上,不仅仅是javascript,在很多语言中都...

    oogh 评论0 收藏0

发表评论

0条评论

MageekChiu

|高级讲师

TA的文章

阅读更多
最新活动
阅读需要支付1元查看
<