1.背景介绍

任何程序设计语言都有作用域的概念,简单的说,作用域就是变量与函数的可访问范围,即作用域控制着变量与函数的可见性和生命周期。在JavaScript中,变量的作用域有全局作用域和局部作用域两种。

2.知识剖析

1.全局作用域(Global Scope)

在代码中任何地方都能访问到的对象拥有全局作用域,一般来说一下几种情形拥有全局作用域:

(1)最外层函数和在最外层函数外面定义的变量拥有全局作用域,例如:

            
var aName="泷泽萝拉";
function doSomething(){var bName="吉泽明步";function innerSay(){alert(bName);}innerSay();
}
alert(aName); //泷泽萝拉
alert(bName); //脚本错误
doSomething(); //吉泽明步
innerSay() //脚本错误

(2)所有末定义直接赋值的变量自动声明为拥有全局作用域,例如:

            
function doSomething(){bName="加藤鹰";alert(bName)
}
alert(bName); //加藤鹰

(3)所有window对象的属性拥有全局作用域

一般情况下,window对象的内置属性都都拥有全局作用域,例如window.name、window.top等等。

1. 局部作用域(Local Scope)

和全局作用域相反,局部作用域一般只在固定的代码片段内可访问到,最常见的例如函数内部,所有在一些地方也会看到有人把这种作用域成为函数作用域

例如下列代码中的bName和函数innerSay都只拥有局部作用域:

            
function doSomething(){var bName="杨泽平";function innerSay(){alert(bName);}innerSay();
}
alert(bName); //脚本错误
innerSay(); //脚本错误

作用域链(Scope Chain)

有了JavaScript的作用域的划分,那么可以将JavaScript的访问作用域连成一个链式树状结构.
JavaScript的作用域链一旦能清晰的了解,那么对于JavaScript的变量与闭包就是非常清晰的了.
下面采用绘图的办法,绘制作用域链.
3.1 绘制规则:
1) 作用域链就是对象的数组
2) 全部script是0级链,每个对象占一个位置
3) 凡是看到函数延伸一个链出来,一级级展开
4) 访问首先看当前函数,如果没有定义往上一级链检查
5) 如此往复,直到0级链

先看一段代码

                var num = 10;var func1 = function() {var num = 20;var func2 = function() {var num = 30;alert(num);};func2();};var func2 = function() {var num = 20;var func3 = function() {alert(num);};func3();};func1();func2();

下面分析一下这段代码:
-> 首先整段代码是一个全局作用域,可以标记为0级作用域链,那么久有一个数组
var link_0 = [ num, func1, func2 ];// 这里用伪代码描述
-> 在这里func1和func2都是函数,因此引出两条1级作用域链,分别为
var link_1 = { func1: [ num, func2 ] };// 这里用伪代码描述
var link_1 = { func2: [ num, func3 ] };// 这里用伪代码描述
-> 第一条1级链衍生出2级链
var link_2 = { func2: [ num ] };// 这里用伪代码描述
-> 第二条1级链中没有定义变量,是一个空链,就表示为
var link_2 = { func3: [ ] };

-> 将上面代码整合一下,就可以将作用域链表示为

               // 这里用伪代码描述var link = [ // 0级链num,{ func1 : [ // 第一条1级链num,{ func2 : [ // 2级链num] }]},{ func2 : [ // 第二条1级链num,{ func3 : [] }]}];

4.常见问题

如何更加直观的体现作用域链

4.解决方案

脚本之家

5.编码实战

6.扩展思考

如何运用作用域链的知识进行闭包优化优化

脚本之家

其实作用域链就是JS引擎查询数据的一个链表,后定义的覆盖先定义的,查询不到定义的数据就往深一层查询,一直到全局作用域为止

                
function changeColor(){document.getElementById("btnChange").onclick=function(){document.getElementById("targetCanvas").style.backgroundColor="red";};
}

这段代码可以重写如下:

                
function changeColor(){var doc=document;doc.getElementById("btnChange").onclick=function(){doc.getElementById("targetCanvas").style.backgroundColor="red";};
}

这段代码比较简单,但是如果程序中有大量的全局变量被从反复访问,那么重写后的代码性能会有显著改善。

7.参考文献

参考一51cto

参考二博客

问题:

//undefined和 is a not defined有什么区别??

==》当声明一个变量而没有赋值的时候会是undefined,当根被没有声明变量的时候,会是 is a not defined。

//那为什么不能都设置成全局变量呢???

==》全局变量其实都是在Window对象下的变量,设置的全局多了以后会造成全局变量污染,不容易维护,且性能降低。。。每找一次变量,找到最外层window才能找到变量。

//在函数中途声明一个变量,之前的还是局部变量么?

==》在函数中途声明一个变量,这个声明会有一个声明提升的作用,相当于把 var 提升到函数的最上部,所以这个变量就是局部变量了。

//作用域和执行环境一样么??

==》是两个不同的概念。作用域主要指的就是作用域就是变量与函数的可访问范围,即作用域控制着变量与函数的可见性和生命周期,而执行环境是定义了变量或者函数有权访问的其他数据,决定了各自的 行为。

//为什么window.xx也能找到属性名字???

==》因为全局变量本身就是在window对象下的。

就行像调用window.location等其他方法一样。

//怎么让一个函数自执行

==>         (function () {}())       即可自执行。

如何理解JS中作用域和作用域链相关推荐

  1. 深入理解JS中的变量作用域

    在JS当中一个变量的作用域(scope)是程序中定义这个变量的区域.变量分为两类:全局(global)的和局部的.其中全局变量的作用域是全局性的,即在JavaScript代码中,它处处都有定义.而在函 ...

  2. 如何更加简单的理解JS中的原型原型链概念

    前面写了很多关于前端经验之谈,今天就来点干货吧.这篇文章将会介绍原型这个概念 原型是整个Javascript中比较重要的概念,如果面向对象想要学好,那么这个东西你必须要了解,不然后面的原型链,继承,多 ...

  3. js原型和原型链_理解JS中的原型和原型链

    导读:JavaScript中(JS)的原型和原型链是web前端开发面试中经常被问到的问题:同时,如果我们能很好的理解JS中的原型和原型链,对于控制台输出的很多信息我们也能更好的理解,而原型链也是实现继 ...

  4. 深入理解Js中的this

    深入理解Js中的this JavaScript作用域为静态作用域static scope,但是在Js中的this却是一个例外,this的指向问题就类似于动态作用域,其并不关心函数和作用域是如何声明以及 ...

  5. 理解js中的面向对象

    目录 前言: 一点疑问: 1.封装 2.继承 原型链的查找机制 不容易理解的点: ----重点在最后---- 前言: js是一门面向对象的语言,但是又没有类的概念,虽然后来加入了class,但也就是个 ...

  6. 彻底理解js中this

    相关博文:http://blog.csdn.net/libin_1/article/details/49996815 彻底理解js中this的指向,不必硬背. 首先必须要说的是,this的指向在函数定 ...

  7. 理解js中this的指向

    彻底理解js中this的指向 JavaScript 的 this 指向问题深度解析 转载于:https://www.cnblogs.com/jeacy/p/6509616.html

  8. 理解JS中的声明式与命令式

    理解JS中的声明式与命令式? 声明式编程 :告诉机器你想要的是什么    让机器想出去做 优缺点 : 声明式减少了可变量(Immutable Variable)的声明,程序更为安全,   代码更加简洁 ...

  9. js中原型,原型链,原型链继承的个人理解

    一.什么是原型? 每一个javascript对象在创建的时候就会有一个与之关联的对象B产生,对象B就是所说的"原型". 1)原型也是一个对象,其他对象可以通过原型实现属性继承, 2 ...

最新文章

  1. 一种注册表沙箱的思路、实现——研究Reactos中注册表函数的实现2
  2. sqlserver导入execl数据ACE.OLEDB.12.0错误
  3. Forrester报告:人工智能将取代6%的工作岗位
  4. 如何通过网页方式将jar包上传到nexus?
  5. 软件评测师32小时-第一小时 软件测试概论
  6. dsp中C语言线性缓冲,TI C64x+ DSP CACHE 一致性分析与维护
  7. 实用的图片批量压缩优化工具
  8. 通过OPENSSL建立证书以及CSR证书签名过程
  9. list去重和list倒叙
  10. java读txt文件乱码_java读取txt文件时出现中文乱码怎么解决
  11. css如何去掉图片里面存在的背景色
  12. SRA数据下载(通过EBI-ENA数据库,使用ASpera)
  13. 2021年最详细的Android屏幕适配方案汇总
  14. 2018富途证券前端实习面试总结
  15. springboot 全局时间转换器
  16. 一文看懂:如何将小程序分享到朋友圈[建议收藏]
  17. 滴滴夜莺:从监控告警系统向运维平台演化
  18. 三步教你安装微软系统,使用uiso制作U盘启动盘
  19. 满头黑发开始_python
  20. SpringCloud分布式微服务搭建(一)

热门文章

  1. 电子烟出口英国需要什么认证?
  2. 【Python】DataFrame使用drop_duplicates()函数去重(不)保留重复值,取重复值
  3. matlab 高斯一阶导,高斯函数及其各阶导数
  4. Jira各种流程状态
  5. Thumbnails的使用
  6. REID triplet loss 图片按照N * K 进行预处理
  7. 通过这次疫情,互联网行业会给我们带来哪些新的发展趋势与机遇?
  8. 使用css3实现图片旋转
  9. 沈阳师范大学-PTA-数据结构- jmu-ds-最长数字序列
  10. Mac地址;Mac地址的分类