javascript闭包的理解和实例


Posted in Javascript onAugust 12, 2010

顺便提示一下:
词法作用域:变量的作用域是在定义时决定而不是执行时决定,也就是说词法作用域取决于源码,通过静态分析就能确定,因此词法作用域也叫做静态作用域。 with和eval除外,所以只能说JS的作用域机制非常接近词法作用域(Lexical scope)。

下面是一个简单的使用全局变量的闭包实例:

var sWord="Hello,Welcome to web前端开发工程师的博客,请多多指教。" 
function disWord(){ 
alert(sWord); 
} 
disWord();

解析:脚本载入到内存的时候,disWord并没有计算sWord的值,而是函数disWord调用的时候执行了sWord的计算。

下面是函数中定义另一个函数的闭包实例:

var iNum=10; 
function add(num1,num2){ 
function doAdd(){return num1+num2+iNum;} 
return doAdd(); 
}

解析:内部函数doAdd是个闭包,它将获取传入参数num1,num2和全局变量iNum的值,doAdd不接受参数,add最后一步调用doAdd,请两个参数和全局变量求和返回,可以看得出doAdd使用的值是在执行环境中获得的。

下面是在网上找的几个例子,理解词法作用域和闭包

、案例一 
/*全局(window)域下的一段代码*/ 
function a(i) { 
var i; 
alert(i); 
}; 
a(10);

疑问:上面的代码会输出什么呢?
答案:10。
具体执行过程
a 函数有一个形参 i,调用 a 函数时传入实参 10,形参 i=10
接着定义一个同名的局部变量 i,未赋值
alert 输出 10
思考:局部变量 i 和形参 i 是同一个存储空间吗?
、案例二

1 /*全局(window)域下的一段代码*/ 
2 function a(i) { 
3 alert(i); 
4 alert(arguments[0]); //arguments[0]应该就是形参 i 
5 var i = 2; 
6 alert(i); 
7 alert(arguments[0]); 
8 }; 
9 a(10);

疑问:上面的代码又会输出什么呢?
答案:10,10,2,2
具体执行过程
函数有一个形参i,调用 a 函数时传入实参 10,形参 i=10
第一个 alert 把形参 i 的值 10 输出
第二个 alert 把 arguments[0] 输出,应该也是 i
接着定义个局部变量 i 并赋值为2,这时候局部变量 i=2
第三个 alert 就把局部变量 i 的值 2 输出
第四个alert再次把 argumentsa[0] 输出
思考:这里能说明局部变量 i 和形参 i 的值相同吗?

、案例三

/*全局(window)域下的一段代码*/ 
function a(i) { 
var i = i; 
alert(i); 
}; 
a(10)

疑问:上面的代码又又会输出什么呢?
答案:10
具体执行过程
第一句声明一个与形参 i 同名的局部变量 i,根据结果我们知道,后一个 i 是指向了
形参 i,所以这里就等于把形参 i 的值 10 赋了局部变量 i
第二个 alert 当然就输出 10
思考:结合案列二,这里基本能说明局部变量 i 和形参 i 指向了同一个存储地址!
、案例四
/*全局(window)域下的一段代码*/ 
var i=10; 
function a() { 
alert(i); 
var i = 2; 
alert(i); 
}; 
a();

疑问:上面的代码又会输出什么呢?
答案:undefined, 2
具体执行过程
第一个alert输出undefined
第二个alert输出 2
思考:到底怎么回事儿?
看到上面的几个例子,你可能会想到底是怎么执行的呢?执行的细节又是怎么样的呢? JS 引擎的工作方式是怎样的呢?
解析过程
、执行顺序
编译型语言,编译步骤分为:词法分析、语法分析、语义检查、代码优化和字节生成。
解释型语言,通过词法分析和语法分析得到语法分析树后,就可以开始解释执行了。这里是一个简单原始的关于解析过程的原理,仅作为参考,详细的解析过程(各种JS引擎还有不同)还需要更深一步的研究
javascript的执行过程,如果一个文档流中包含多个script代码段(用script标签分隔的js代码或引入的js文件),它们的运行顺序是:
步骤1. 读入第一个代码段(js执行引擎并非一行一行地执行程序,而是一段一段地分析执行的)
步骤2. 做词法分析和语法分析,有错则报语法错误(比如括号不匹配等),并跳转到步骤5
步骤3. 对【var】变量和【function】定义做“预解析“(永远不会报错的,因为只解析正确的声明)
步骤4. 执行代码段,有错则报错(比如变量未定义)
步骤5. 如果还有下一个代码段,则读入下一个代码段,重复步骤2
步骤6. 结束
、特殊说明
全局域(window)域下所有JS代码可以被看成是一个“匿名方法“,它会被自动执行,而此“匿名方法“内的其它方法则是在被显示调用的时候才被执行
、关键步骤
上面的过程,我们主要是分成两个阶段
解析:就是通过语法分析和预解析构造合法的语法分析树。
执行:执行具体的某个function,JS引擎在执行每个函数实例时,都会创建一个执行环境(ExecutionContext)和活动对象(activeObject)(它们属于宿主对象,与函数实例的生命周期保持一致)
在这里有更详细的实例分析资料:https://3water.com/article/24547.htm
Javascript 相关文章推荐
In Javascript Class, how to call the prototype method.(three method)
Jan 09 Javascript
js escape,unescape解决中文乱码问题的方法
May 26 Javascript
js实现键盘操作实现div的移动或改变的原理及代码
Jun 23 Javascript
JS函数的定义与调用方法推荐
May 12 Javascript
jQuery实现将div中滚动条滚动到指定位置的方法
Aug 10 Javascript
Angular2 PrimeNG分页模块学习
Jan 14 Javascript
bootstrap weebox 支持ajax的模态弹出框
Feb 23 Javascript
angularjs $http实现form表单提交示例
Jun 09 Javascript
详解webpack4多入口、多页面项目构建案例
May 25 Javascript
layui 动态设置checbox 选中状态的例子
Sep 02 Javascript
小程序按钮避免多次调用接口和点击方案实现(不用showLoading)
Apr 15 Javascript
浅谈JavaScript作用域
Dec 06 Javascript
javascript 词法作用域和闭包分析说明
Aug 12 #Javascript
判断客户端浏览器是否安装了Flash插件的多种方法
Aug 11 #Javascript
基于JQuery的数字改变的动画效果--可用来做计数器
Aug 11 #Javascript
JQuery最佳实践之精妙的自定义事件
Aug 11 #Javascript
用js解决数字不能换行问题
Aug 10 #Javascript
JavaScript 错误处理与调试经验总结
Aug 10 #Javascript
根据对象的某一属性进行排序的js代码(如:name,age)
Aug 10 #Javascript
You might like
php实现解析xml并生成sql语句的方法
2018/02/03 PHP
IE与FireFox的兼容性问题分析
2007/04/22 Javascript
JS的递增/递减运算符和带操作的赋值运算符的等价式
2007/12/08 Javascript
不安全的常用的js写法
2009/09/15 Javascript
简洁短小的 JavaScript IE 浏览器判定代码
2010/03/21 Javascript
javascript中使用css需要注意的地方小结
2010/09/01 Javascript
javascript进行数组追加方法小结
2014/06/16 Javascript
jQuery实现冻结表格行和列
2015/04/29 Javascript
简介JavaScript中POSITIVE_INFINITY值的使用
2015/06/05 Javascript
BootStrap智能表单实战系列(九)表单图片上传的支持
2016/06/13 Javascript
判断输入的字符串是否是日期格式的简单方法
2016/07/11 Javascript
基于jQuery实现页面搜索功能
2020/03/26 Javascript
给easyui datebox扩展一个清空的实例
2016/11/09 Javascript
通过AngularJS实现图片上传及缩略图展示示例
2017/01/03 Javascript
微信小程序 缓存(本地缓存、异步缓存、同步缓存)详解
2017/01/17 Javascript
解析Vue2.0双向绑定实现原理
2017/02/23 Javascript
Bootstrap免费字体和图标网站(值得收藏)
2017/03/16 Javascript
详解JS中的attribute属性
2017/04/25 Javascript
浅谈vue中改elementUI默认样式引发的static与assets的区别
2018/02/03 Javascript
vue组件与复用详解
2018/04/08 Javascript
Angular4.x Event (DOM事件和自定义事件详解)
2018/10/09 Javascript
详解vue-router数据加载与缓存使用总结
2018/10/29 Javascript
layui监听单元格编辑前后交互的例子
2019/09/16 Javascript
JavaScript代理模式原理与用法实例详解
2020/03/10 Javascript
[01:52]2014DOTA2西雅图邀请赛 V社开大会你不知道的小秘密
2014/07/08 DOTA
树莓派使用USB摄像头和motion实现监控
2019/06/22 Python
Betsey Johnson官网:妖娆可爱的连衣裙及鞋子、手袋和配件
2016/12/30 全球购物
高中生期末评语
2014/01/28 职场文书
西式婚礼主持词
2014/03/13 职场文书
廉洁家庭事迹材料
2014/05/15 职场文书
毕业实习自我鉴定范文2014
2014/09/26 职场文书
学生会主席任命书
2015/09/21 职场文书
有关保护环境的宣传标语100条
2019/08/07 职场文书
漫画「日和酱的要求是绝对的」第3卷封面公开
2022/03/21 日漫
室外天线与收音机天线杆接合方法
2022/04/05 无线电
讨论nginx location 顺序问题
2022/05/30 Servers