一次让你了解全部JavaScript的作用域
前言
作用域决定了变量的生命周期和可见性,变量在作用域范围之外是不可见的。
JavaScript的作用域包括:模块作用域,函数作用域,块作用域,词法作用域和全局作用域。
全局作用域
在任何函数、块或模块范围之外定义的变量具有全局作用域。可以在程序的任意位置访问全局变量。
当启用模块系统时,创建全局变量会变得困难,但仍然可以做到这一点。可以在HTML中定义一个变量,这个变量需要在函数之外声明,这样就可以创建一个全局变量:
console.log(GLOBAL_DATA);
当没有模块系统时,创建全局变量会容易很多。在任何文件中的函数外声明的变量都是全局变量。
全局变量贯穿于程序的整个生命周期。
另一种创建全局变量的方法是在程序的任意位置使用window全局对象:
window.GLOBAL_DATA={value:1};
这样GLOBAL_DATA变量会随处可见。
console.log(GLOBAL_DATA)
不过你也知道这种做法是不好的。
模块作用域
如果不启用模块,在所有函数之外声明的变量是全局变量。在模块中,在函数外部声明的变量都是隐藏的,除非显式导出,否则不可用于其他模块。
导出使函数或对象可用于其他模块。在这个例子中,我从模块文件sequence.js中导出了一个函数:
//insequence.js export{sequence,toList,take};
当前模块可以通过导入来使用其他模块的函数或对象成。
import{sequence,toList,toList}from"./sequence";
在某种程度上,我们可以认为模块是一个自动执行的函数,它将import的数据作为输入,然后返回export的数据。
函数作用域
函数作用域意味着在函数中定义的参数和变量在函数内的任何位置都可见,但是在函数外部不可见。
下面是一个自动执行的函数,被称为IIFE。
(functionautoexecute(){ letx=1; })(); console.log(x); //UncaughtReferenceError:xisnotdefined
IIFE的意思是立即调用函数表达式,是一个在定义后立即运行的函数。
用var声明的变量只有函数作用域。更重要的是,用var声明的变量被提升到其作用域的顶部。通过这种方式,可以在声明之前访问它们。看看下面的代码:
functiondoSomething(){ console.log(x); varx=1; } doSomething();//undefined
这种事不会发生在let中。用let声明的变量只能在定义后访问。
functiondoSomething(){ console.log(x); letx=1; } doSomething(); //UncaughtReferenceError:xisnotdefined
用var声明的变量可以在同一作用域下多次重新声明:
functiondoSomething(){ varx=1 varx=2; console.log(x); } doSomething();
用let或const声明的变量不能在同一作用域内重新声明:
functiondoSomething(){ letx=1 letx=2; } //UncaughtSyntaxError:Identifier'x'hasalreadybeendeclared
也许我们可以不必关心这一点,因为var已经开始变得过时了。
块作用域
块作用域用花括号定义。它由{和}分隔。
用let和const声明的变量可以受到块作用域的约束,只能在定义它们的块中访问。
思考下面这段关于let块范围的代码:
letx=1; { letx=2; } console.log(x);//1
相反,var声明不受块作用域的约束:
varx=1; { varx=2; } console.log(x);//2
另一个常见问题是在循环中使用类似setTimeout()的异步操作。下面的循环代码将显示五次数字5。
(functionrun(){ for(vari=0;i<5;i++){ setTimeout(functionlogValue(){ console.log(i);//5 },100); } })();
带有let声明的for循环语句在每次循环都会创建一个新的变量并设置到块作用域。下一段循环代码将会显示012345。
(functionrun(){ for(leti=0;i<5;i++){ setTimeout(functionlog(){ console.log(i);//01234 },100); } })();
词法作用域
词法作用域是内部函数访问定义它的外部作用域的能力。
看一下这段代码:
(functionautorun(){ letx=1; functionlog(){ console.log(x); }; functionrun(fn){ letx=100; fn(); } run(log);//1 })();
log函数是一个闭包。它从父函数autorun()引用x变量,而不是run()函数中的x变量。
闭包函数可以访问创建它的作用域,而不是它自己的作用域。
autorun()的局部函数作用域是log()函数的词法作用域。
作用域链
每个作用域都有一个指向父作用域的链接。当使用变量时,JavaScript会向下查看作用域链,直到它找到所请求的变量或者到达全局作用域(即作用域链的末尾)。
看下面这个例子:
letx0=0; (functionautorun1(){ letx1=1; (functionautorun2(){ letx2=2; (functionautorun3(){ letx3=3; console.log(x0+""+x1+""+x2+""+x3);//0123 })(); })(); })();
内部函数autorun3()可以访问本地x3变量。还可以从外部函数访问变量x1和x2和全局变量x0。
如果找不到变量,它将在严格模式下返回错误。
"usestrict"; x=1; console.log(x) //UncaughtReferenceError:xisnotdefined
非严格模式也被称为“草率模式”,它会草率的创建一个全局变量。
x=1; console.log(x);//1
总结
在全局作用域中定义的变量可在程序的任何位置使用。
在模块中,在函数外部声明的变量都是隐藏的,除非被显式导出,否则不可用于其他模块。
函数作用域意味着函数中定义的参数和变量在函数的任意位置都可见
用let和const声明的变量具有块作用域。var没有块作用域。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持毛票票。
声明:本文内容来源于网络,版权归原作者所有,内容由互联网用户自发贡献自行上传,本网站不拥有所有权,未作人工编辑处理,也不承担相关法律责任。如果您发现有涉嫌版权的内容,欢迎发送邮件至:czq8825#qq.com(发邮件时,请将#更换为@)进行举报,并提供相关证据,一经查实,本站将立刻删除涉嫌侵权内容。