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)已经有一段时间了.我之前只是学会了如何使用它们 ...
随机推荐
- leetcode748
public class Solution { public string ShortestCompletingWord(string licensePlate, string[] words) { ...
- UI5-文档-4.2-Bootstrap
在使用SAPUI5做一些事情之前,我们需要加载并初始化它.加载和初始化SAPUI5的过程称为引导.一旦引导完成,我们只需显示一个警告. Preview An alert "UI5 is re ...
- eclipse 注释字体不一致的问题
eclipse中 1.解决注释的文字大小不一的情况 2.想让注释和代码大小不一样 3.win10系统下,设置Text Font时找不到Courier New字体 1.解决注释的文字大小不一的情况 打开 ...
- HIBERNATE知识复习记录2-继承关系
发现了一篇和我类似的学习尚硅谷视频写的文章,内容如下,比我说的详细全面,可以看一下: [原创]java WEB学习笔记87:Hibernate学习之路-- -映射 继承关系(subclass , jo ...
- Unified shader model
https://en.wikipedia.org/wiki/Unified_shader_model In the field of 3D computer graphics, the Unified ...
- 不用登陆密码也能进路由器,适用于TP、磊科、腾达
结合wooyun提供的腾达COOKIE漏洞,结合自己的经验,成功进入腾达路由器破解其登陆密码和无线密码. 教程开始: 所用工具:WebCruiser 输入路由网关,出现登陆界面. 选择:COOKIE ...
- intellij idea 的安装与简单使用
1.将安装包拷贝到指定目录,特别注意不要有中文路径和空格,路径不要太深 2.点击安装(如果是win10系统要使用管理员权限安装) 3. 4.修改默认安装目录:一般来说我们都不要把软件安装在 ...
- HTML的属性和css基础
1.name属性: name属性,用于指定标签元素的名称,<a>标签内必须提供href或name属性:<a name ="value"> 2.id属性: 1 ...
- Access to the path 'C:\inetpub\wwwroot\mysite\images\savehere' is denied.
访问路径被拒绝 我解决了这个设置: IIS>应用程序池> [您的站点]>高级设置...>标识>内置帐户> LocalSystem
- iOS下JS与OC互相调用(七)--Cordova 环境搭建
Cordova大家可能比较陌生,但肯定听过 PhoneGap ,Cordova 就是 PhoneGap 被 Adobe 收购后所改的名字.它是一个可以让 JS 与原生代码互相通信的一个库,并且提供了一 ...