JavaScript 闭包的例子
例子出自<<JavaScript权威指南>>, 加上个人的理解和总结, 欢迎交流!
/*****************************************************************************/
/* 将一个立即调用函数的返回值赋给变量uniqueInteger
* 特点: 内部函数可以访问外部函数中的局部变量,当外部匿名函数立即调用返回之后, 除了返回的函数对象外
* 其他任何代码都无法访问私有变量counter, 实现了变量的隔离(相当于命名空间).
*/
var uniqueInteger = (function(){
var counter = ;
return function(){
return counter++;
};
}());
/*****************************************************************************/ /*****************************************************************************/
/* 多个嵌套函数共享一个私有变量
*/
function counter(){
var n = ;
return {
count: function(){
return n++;
},
reset: function(){
n = ;
}
};
}
var c = counter(); // 相当于上一个例子中的立即调用, 只是得到的是一个直接对象, 而不是函数对象
var d = counter(); // 定义了另一个对象. 和c各自独立.
// 两个对象各自维护自己的私有变量 n , 其他外部代码不能再直接访问它们的 n 的值.
console.log(c.count()); //0
console.log(c.count()); //1
c.reset();
console.log(c.count()); //0
d.count(); d.reset();
//上面的counter()也可以类似如下实现, 因为参数和局部变量一样保存在作用域链中.
function counter2(n){
n = n || ;
return {
get count(){return n++;},
set count(m){
if(m >= n) n = m;
else throw Error("count can only be set to a larger value");
}
};
}
var e = counter2();
e.count;
console.log(e.count); // 11
e.count = ;
console.log(e.count); // 20
console.log(e.count); // 21
/*****************************************************************************/ /*****************************************************************************/
/* 利用闭包实现私有属性存取器方法
* 为传入的某个对象加入setter和getter方法, 但是setter/getter方法操作的属性并没有添加到
* 对象 o 中, 而是由闭包代为保管!
* 这样只能通过setter/getter进行访问, 其他地方无法直接访问,成为私有变量.
*/
function addPrivateProperty(o,name,predicate){
var value;
o['get'+name] = function(){
return value;
},
o['set'+name] = function(v){
if(predicate && !predicate(v))
throw Error("set"+name+":invalid value "+v);
else
value = v;
};
}
var o = {};
addPrivateProperty(o,"Name",function(x){return typeof(x) == "string";});
o.setName("roger");
console.log(o.getName()); // roger
/*****************************************************************************/ /*****************************************************************************/
/* 当多个闭包共享同样的私有变量或变量时要注意作用域时动态变化的.
* 嵌套的函数不会讲作用域内的私有成员复制一份(只是引用), 也不会对所绑定的变量生成静态快照.
*/
function constfunc(v){
return function(){
return v;
};
}
var funcs = [];
for(var i=;i<;++i){
// 创建多个闭包共享变量 i.作用域链在变化,直到循环结束, 但是执行constfunc时保存了i的状态.
funcs[i] = constfunc(i);
}
console.log(funcs[]()); // 5
function constfunc2(){
var funcs = [];
for(var i=;i<;++i){
//函数对象并没有保存i的状态, 在外面调用时作用域的状态已经变化了, 和预期不一样.
//因为内嵌对象不会将作用域内的私有成员复制一份(只是一份引用而已)
funcs[i] = function(){
return i;
};
}
return funcs;
}
var funcs = constfunc2();
console.log(funcs[]()); //10
function constfunc3(){
var funcs = [];
for(var i=;i<;++i){
funcs[i] = (function(){
return i;
}()); //使用函数立即调用, 直接将闭包的状态保存在了funcs[i]中.
}
return funcs;
}
var funcs = constfunc3();
console.log(funcs[]); // 5
/*****************************************************************************/ /*****************************************************************************/
/*需要注意this和arguments
1. this不是变量, 而是关键字, 会随着作用域而变化
2. arguments是函数在调用时引擎自动创建的变量, 也是随作用域而变化的
解决方法: 将外部函数的this和arguments的内容(而不是引用)保存到局部变量中,
从而避免作用域变化带来的不确定性
*/
-->
JavaScript 闭包的例子的更多相关文章
- javascript闭包学习例子
javascript中的闭包个很让人头疼的概念.总结一下 闭包是指有权访问一个函数作用域中的变量的函数.创建闭包最常见的方式,是在一个函数内部创建另一个函数,用return返回出去. 使用闭包可能造成 ...
- javascript 闭包理解例子
function Jquery(){ this.name = 'ysr'; this.sex = 'man'; return { x: this, age : 26 } } var b = new J ...
- 《Web 前端面试指南》1、JavaScript 闭包深入浅出
闭包是什么? 闭包是内部函数可以访问外部函数的变量.它可以访问三个作用域:首先可以访问自己的作用域(也就是定义在大括号内的变量),它也能访问外部函数的变量,和它能访问全局变量. 内部函数不仅可以访问外 ...
- JavaScript 闭包深入浅出
闭包是什么? 闭包是内部函数可以访问外部函数的变量.它可以访问三个作用域:首先可以访问自己的作用域(也就是定义在大括号内的变量),它也能访问外部函数的变量,和它能访问全局变量. 内部函数不仅可以访问外 ...
- javascript闭包和作用域链
最近在学习前端知识,看到javascript闭包这里总是云里雾里.于是翻阅了好多资料记录下来本人对闭包的理解. 首先,什么是闭包?看了各位大牛的定义和描述各式各样,我个人认为最容易一种说法: 外部函数 ...
- JavaScript闭包模型
JavaScript闭包模型 ----- [原创翻译]2016-09-01 09:32:22 < 一> 闭包并不神秘 本文利用JavaScript代码来阐述闭包,目的是为了使普通 ...
- javascript 闭包(转)
一.变量的作用域 要理解闭包,首先必须理解Javascript特殊的变量作用域. 变量的作用域无非就是两种:全局变量和局部变量. Javascript语言的特殊之处,就在于函数内部可以直接读取全局变量 ...
- 最简明的JavaScript闭包解释
最简明的JavaScript闭包解释 JavaScript是这几年最火的编程语言之一,从前端到服务器端,再到脚本,好像没有一个地方没有JavaScript的身影.这个世界上任何的一种事物的存在必然有其 ...
- JavaScript闭包的底层运行机制
转自:http://blog.leapoahead.com/2015/09/15/js-closure/ 我研究JavaScript闭包(closure)已经有一段时间了.我之前只是学会了如何使用它们 ...
随机推荐
- jpa-spring -basic
applicationContent.xml <?xml version="1.0" encoding="UTF-8"?> <beans xm ...
- myBatis连接MySQL报异常:No operations allowed after connection closed.Connection was implicitly closed
网站运行一个晚上,早上来上班,发现报错: ### Error querying database. Cause: com.mysql.jdbc.exceptions.jdbc4.MySQLNonTra ...
- xe7 Unresolved external CSPIN.OBJ
工程里打开使用了CSPIN控件的界面窗体,再编译就好了 [ilink32 Error] Error: Unresolved external 'Vcl::Controls::TControl::Set ...
- VBA 禁止保存
禁止保存 在workbook事件中 Private Sub Workbook_BeforeClose(Cancel As Boolean) Me.Saved = TrueEnd Sub Priv ...
- jsfl 第一天
ctrl+f10,打开操作记录 通过打开到目标fla的舞台,然后运行写好的jsfl,默认就以目标fla为调试对象. 通过name可以获取层名字,帧标签,元件实例名等,name属性要根据对象的应用而产生 ...
- mysql 5.7.10使用dbforget Studio 连接异常
提示:The 'INFORMATION_SCHEMA.SESSION_VARIABLES' feature is disabled; see the documentation for 'show_c ...
- vue深入了解组件——插槽
一.插槽内容 Vue实现了一套内容分发的API,这套API基于当前的Web Components规范草案,将 <slot> 元素作为承载分发的内容的出口. 它允许你像这样合成组件: &l ...
- box2d 计算下一帧的位置/角度
var dt:Number=1/30; var y0:Number=_body.GetPosition().y; var y:Number=y0+_body.GetLinearVelocity().y ...
- Eclipse中Ctrl+Shift+f快捷键无效的解决方式
某天突然发现idea非常重要的快捷键ctrl+shift+f无效了,网上搜了很多都说是qq快捷键冲突,但是找了下qq快捷键却没有解决,现在给大家一个解决快捷键冲突的思路: 1.查看QQ快捷键--> ...
- 3.3 JSP内置对象概述
1.request 1.1 request获取页面传来的参数,参数通过浏览器网址和后面添加?的方式传达. 传参:”show.jsp?id=001” 获取参数:request.getParameter( ...