变量所在的上下文,指的是变量在哪些地方能够访问前端
对于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中有函数做用域,指的是做用域在函数内部。这里一共说了三种做用域,其实能够说是两种:一种是全局做用域,而是局部做用域(函数做用域、块级做用域),块级做用域概念又包括了函数做用域。code
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()函数做用域下的变量。对象
做用域链的前端始终是当前环境做用域下变量对象,逐层往外做用域连接,最后端是全局变量环境下的变量,这些变量时连接在一块儿,在解析一个变量时从链前端日后端搜索(从内不找外部找),可是有一点值得注意:每一个变量的做用域老是从自身声明的做用域往外找,而不是调用它的地方 :ip
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吗?io
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() //输出多少?
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()后再进行的赋值,那么这里应该是多少呢?console
var a
时,也就是执行到了fn3()代码前,函数声明和变量声明会提早至代码前端,因此这里声明并无影响到输出值得改变,可是赋值操做是按照程序顺序执行的,当调用前,a只声明没有赋值,则会输出undefined。 而具体变量查找是符合做用域链的顺序来的.总结以下:
我的学习备忘,若有谬误,欢迎指正。