所谓的作用域,可以简单理解为一个可以读、写的范围(区域),有些js经验的同学可能会说:"js没有块级作用域",js除了全局作用域外,只有函数可以创建作用域。作用域的一个好处就是可以隔离变量。
我们通过一些例子来帮助我们理解js中的作用域。
alert(a); var a = 1;
如果对作用域一点不了解的同学可能会说 alert的是1或者报错;但实际上是undefined;
说到这里,我们首先说一下js逐行解析代码之前做的一些准备工作,
js在逐行读代码之前,会做一些“预解析”工作,会先提前找到一些”小东西”,当然”js解析器“不会随便找一些数据的,它会根据var,function,参数来找。
”js解析器“它比较”懒“,在正式运行代码之前都会给var声明的变量赋值为undefined,也就是var a = undefined;会把整个函数看作一个代码块,不去管里边有多少代码。参数等到后边例子中会说。
当所有准备工作都做好后,“JS解析器”就开始逐行执行代码了,现在我们来分析开始的这个例子就很容易明白为什么是undefined了。
再来看下边这个例子
alert(a); var a = 1; alert(a); var a = 2; alert(a);
我们来一点点分析这个
首先 ”预解析“: 解析器会找var
读到第二行时 a = undefined;
读到第四行时 依然 a = undefined;
正式逐行执行代码:
第一行 alert:undefined
第二行 a = 1;
第三行 alert:1;
第五行 alert:2
接着看下边这个例子
alert(a); var a = 1; alert(a); function a (){ alert(2); } alert(a); var a = 3; alert(a); function a (){ alert(4); } alert(a);
我们依然来一点点分析这个
首先 ”预解析“: 解析器会找var function;
读到第二行时 a = undefined;
读到第四行时 a = function a (){ alert(2);} //所有的函数,在正式运行代码之前,都是整个函数块;变量遇到重名的,只留一个变量,如果变量和函数重名,就只留下函数。
读到第六行时,a = function a (){ alert(2);}
读到第八行时,a = function a (){ alert(4);}
正式逐行执行代码:
第一行 alert: function a (){ alert(4);}
第二行 a = 1; //表达式可以修改预解析的值!
第三行 alert:1;
第四行 函数没有调用,略过;
第五行 alert:1;
第六行 a = 3;
第七行 alert:3
第八行 函数没有调用,略过;
第九行 alert:3
如图所示:
继续看例子:
var a = 1; function fn1(){ alert(a); //undefined var a = 2; } fn1(); alert(a); //1
首先 ”预解析“: 解析器会找var function
读到第一行时 a = undefined;
读到第二行时 fn1 = function fn1 (){alert(2);var a = 2;}
正式逐行执行代码: 第一行 a = 1;
第六行 函数调用,进入函数作用域 在函数作用域内依旧是先预解析,再逐行执行
函数内预解析:a = undefined;
执行:alert:undefined;
a = 2; //此时的a仅为函数作用域中的a,不会影响全局中的a
函数执行完毕,回到全局作用域;
第七行 alert:1;
继续:
var a = 1; function fn1(){ alert(a); //1 a = 2; } fn1(); alert(a); //2
这个例子上边那个例子唯一的区别就是函数中的a没有var,只分析其中关键的地方
在函数作用域中 第三行alert(a),由于函数中没有var a,所以"解析器"会到函数的作用域的上一级作用域去寻找a(作用域上下级关系的确定就看函数是在哪个作用域下创建的,在哪个作用域下创建,就是哪个作用域的下级),此时函数的上一级是全局作用域,在全局作用域中,a = 1,所以此时第三行 alert:1,接着第四行,a = 2赋值,依然是函数作用域中没有a, 所以在上一级作用域,也就是全局作用域中找到a,修改全局作用域中的a, 所以会使全局作用域中的a = 2, 因此第七行 alert:2;
这点要理解清楚,注意有无var的区别。
接着来:
var a = 1; function fn1(a){ alert(a); //undefined a = 2; } fn1(); alert(a); // 1
这个例子和上一个的区别就是多了个参数,参数的作用相当于局部变量,也就是在函数中预解析会有var a = undefined,所以第三行 alert:undefined,第四行 a = 2 改的是函数作用域中的a,不影响全局中的a,第七行alert:1;
接着:
var a = 1; function fn1(a){ alert(a); // 1 a = 2; } fn1(a); alert(a); // 1
这个例子又与上一个有些区别,在第六行函数调用时传了个实参进去,第六行函数实参的a是全局变量a = 1的1,函数执行时,第二行 a = 1,所以第三行alert:1,第七行alert:1。
注意这几个例子之间的区别,别混淆了。
再来一个:
var a = 1; function en(){ var a = 2; fn(); } function fn(){ alert(a); //1 } en();
fn中的a未声明,要到创建这个函数的那个作用域中取值——是“创建”,而不是“调用”这个函数的作用域中。
PS:JavaScript中的作用域和上下文概念
javascript中的作用域(scope)和上下文(context)是这门语言的独到之处,这部分归功于他们带来的灵活性。每个函数有不同的变量上下文和作用域。这些概念是javascript中一些强大的设计模式的后盾。然而这也给开发人员带来很大困惑。下面全面揭示了javascript中的上下文和作用域的不同,以及各种设计模式如何使用他们。
上下文 vs 作用域
首先需要澄清的问题是上下文和作用域是不同的概念。多年来我注意到许多开发者经常将这两个术语混淆,错误的将一个描述为另一个。平心而论,这些术语变得非常混乱不堪。
每个函数调用都有与之相关的作用域和上下文。从根本上说,范围是基于函数(function-based)而上下文是基于对象(object-based)。换句话说,作用域是和每次函数调用时变量的访问有关,并且每次调用都是独立的。上下文总是关键字 this 的值,是调用当前可执行代码的对象的引用。
以上所述是小编给大家介绍的javascript中的作用域(推荐),希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对网站的支持!
js作用域
P70系列延期,华为新旗舰将在下月发布
3月20日消息,近期博主@数码闲聊站 透露,原定三月份发布的华为新旗舰P70系列延期发布,预计4月份上市。
而博主@定焦数码 爆料,华为的P70系列在定位上已经超过了Mate60,成为了重要的旗舰系列之一。它肩负着重返影像领域顶尖的使命。那么这次P70会带来哪些令人惊艳的创新呢?
根据目前爆料的消息来看,华为P70系列将推出三个版本,其中P70和P70 Pro采用了三角形的摄像头模组设计,而P70 Art则采用了与上一代P60 Art相似的不规则形状设计。这样的外观是否好看见仁见智,但辨识度绝对拉满。
更新日志
- 小骆驼-《草原狼2(蓝光CD)》[原抓WAV+CUE]
- 群星《欢迎来到我身边 电影原声专辑》[320K/MP3][105.02MB]
- 群星《欢迎来到我身边 电影原声专辑》[FLAC/分轨][480.9MB]
- 雷婷《梦里蓝天HQⅡ》 2023头版限量编号低速原抓[WAV+CUE][463M]
- 群星《2024好听新歌42》AI调整音效【WAV分轨】
- 王思雨-《思念陪着鸿雁飞》WAV
- 王思雨《喜马拉雅HQ》头版限量编号[WAV+CUE]
- 李健《无时无刻》[WAV+CUE][590M]
- 陈奕迅《酝酿》[WAV分轨][502M]
- 卓依婷《化蝶》2CD[WAV+CUE][1.1G]
- 群星《吉他王(黑胶CD)》[WAV+CUE]
- 齐秦《穿乐(穿越)》[WAV+CUE]
- 发烧珍品《数位CD音响测试-动向效果(九)》【WAV+CUE】
- 邝美云《邝美云精装歌集》[DSF][1.6G]
- 吕方《爱一回伤一回》[WAV+CUE][454M]