javascript 闭包

简介:
原创作品,允许转载,转载时请务必以超链接形式标明文章  原始出处 、作者信息和本声明。否则将追究法律责任。 http://dba10g.blog.51cto.com/764602/1357820

目录

定义

引子

闭包的用途

一个常见错误

 

 

 


定义

闭包是一种特殊的对象。它由两部分构成:函数,以及创建该函数的环境

引子

首先看一个例子

1
2
3
4
5
6
7
8
9
function  makeFunc() {
var  name =  "Mozilla" ;
function  displayName() {
alert(name);
}
return  displayName;
}
var  myFunc = makeFunc();
myFunc();

这段代码看起来别扭却能正常运行。通常,函数中的局部变量仅在函数的执行期间可用。一旦 makeFunc() 执行过后,我们会很合理的认为 name 变量将不再可用。不过,既然代码运行的没问题,显然不是我们想象的那样。在我们的例子中,myFunc 是一个闭包,由 displayName 函数和闭包创建时存在的 "Mozilla" 字符串形成。

这就是闭包,我们在返回函数的时候,也将函数的环境一并返回了。

闭包的用途

  1. 响应事件而执行的函数

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
< html >
< head >
< script  type = "text/javascript" >
function makeSizer(size){
return function(){
document.body.style.fontSize=size+"px";
}
}
var size12= makeSizer(12);
var size14= makeSizer(14);
var size16= makeSizer(16);
window.onload=function(){
document.getElementById("a12").onclick=size12;
document.getElementById("a14").onclick=size14;
document.getElementById("a16"). name="a12" id="a12">12</ a >
< a  name = "a12"  id = "a14" >14</ a >
< a  name = "a12"  id = "a16" >16</ a >
</ body >
</ html >

 2.  模拟似有方法

诸如 Java 在内的一些语言支持将方法声明为私有的,既它们只能被同一个类中的其它方法所调用。

对此,JavaScript 并不提供原生的支持,但是可以使用闭包模拟私有方法。私有方法不仅仅有利于限制对代码的访问:还提供了管理全局命名空间的强大能力,避免非核心的方法弄乱了代码的公共接口部分。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
<script type= "text/javascript" >
function  MyObject(){
var  name= "" ;
return  {
getName: function (){
return  name;
},
setName: function (str){
name=str;
}
}
}
var  mo = MyObject();
mo.setName( "Mo1" );
alert(mo.getName()); //Mo1
alert(mo.name); //undefined
</script>

在上面的例子中,name作为私有属性,getName,setName为公有方法。

 

一个常见的错误

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
< html >
< head >
</ head >
< body >
< p  id = "help" >Helpful notes will appear here</ p >
< p >E-mail: < input  type = "text"  id = "email"  name = "email" ></ p >
< p >Name: < input  type = "text"  id = "name"  name = "name" ></ p >
< p >Age: < input  type = "text"  id = "age"  name = "age" ></ p >
< script  type = "text/javascript" >
function showHelp(help) {
document.getElementById('help').innerHTML = help;
}
function setupHelp() {
var helpText = [
{'id': 'email', 'help': 'Your e-mail address'},
{'id': 'name', 'help': 'Your full name'},
{'id': 'age', 'help': 'Your age (you must be over 16)'}
];
for (var i = 0; i <  helpText.length ; i++) {
var  item  helpText [i];
document.getElementById(item.id) .onfocus  function () {
showHelp(item.help);//item.help 为最后一个对象的help
}
}
}
setupHelp();
</script>
</ body >
</ html >

该问题的原因在于赋给 onfocus 的函数是闭包;它们由函数定义和记录自 setupHelp 函数作用域的环境构成。一共创建了三个闭包,但是它们都共享同一个环境。在 onfocus 的回调被执行时,循环早已经完成,且此时 item 变量(由所有三个闭包所共享)已经指向了 helpText 列表中的最后一项。

 

修正

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
< html >
< head >
</ head >
< body >
< p  id = "help" >Helpful notes will appear here</ p >
< p >E-mail: < input  type = "text"  id = "email"  name = "email" ></ p >
< p >Name: < input  type = "text"  id = "name"  name = "name" ></ p >
< p >Age: < input  type = "text"  id = "age"  name = "age" ></ p >
< script  type = "text/javascript" >
function showHelp(help) {
document.getElementById('help').innerHTML = help;
}
function makeHelpCallback(help){
return function(){
showHelp(help);
}
}
function setupHelp() {
var helpText = [
{'id': 'email', 'help': 'Your e-mail address'},
{'id': 'name', 'help': 'Your full name'},
{'id': 'age', 'help': 'Your age (you must be over 16)'}
];
for (var i = 0; i <  helpText.length ; i++) {
var  item  helpText [i];
document.getElementById(item.id) .onfocus  = makeHelpCallback (item.help);
}
}
setupHelp();
</script>
</ body >
</ html >

这段代码可以如我们所期望的那样工作。所有的回调不再共享同一个环境, makeHelpCallback 函数为每一个回调创建一个新的环境。在这些环境中,help 指向 helpText 数组中对应的字符串。

 

参考:https://developer.mozilla.org/zh-CN/docs/JavaScript/Guide/Closures

 

 

 

本文出自 “简单” 博客,请务必保留此出处http://dba10g.blog.51cto.com/764602/1357820

目录
相关文章
|
23天前
|
JavaScript
闭包(js的问题)
闭包(js的问题)
12 0
|
1月前
|
设计模式 JavaScript 前端开发
js开发:请解释闭包(closure)是什么,以及它的用途。
闭包是JavaScript中的关键特性,允许函数访问并操作外部作用域的变量,常用于实现私有变量、模块化和高阶函数。私有变量示例展示了如何创建无法外部访问的计数器;模块化示例演示了封装私有变量和函数,防止全局污染;高阶函数示例则说明了如何使用闭包创建能接收或返回函数的函数。
16 2
|
1月前
|
存储 缓存 JavaScript
|
1月前
|
自然语言处理 JavaScript 前端开发
探索JavaScript中的闭包:理解其原理与实际应用
探索JavaScript中的闭包:理解其原理与实际应用
19 0
|
1月前
|
自然语言处理 JavaScript 前端开发
深入理解JS的执行上下文、词法作用域和闭包(中)
深入理解JS的执行上下文、词法作用域和闭包(中)
|
1月前
|
存储 自然语言处理 JavaScript
深入理解JS的执行上下文、词法作用域和闭包(上)
深入理解JS的执行上下文、词法作用域和闭包(上)
|
1天前
|
测试技术
js_防抖与节流(闭包的使用)
js_防抖与节流(闭包的使用)
7 0
|
30天前
|
JavaScript 前端开发 Java
深入剖析 JavaScript 闭包
深入探讨JavaScript闭包,了解其定义、特性、优缺点及作用。闭包是函数与其引用环境的组合,允许内层函数访问外层作用域,常驻内存可能导致内存泄露。优点包括创建私有变量,缺点则涉及内存使用。闭包在变量搜索中遵循从内到外的规则,并影响变量的作用域和生存周期。理解闭包有助于优化代码并避免性能问题。
21 1
|
1月前
|
自然语言处理 前端开发 JavaScript
深入理解JavaScript中的闭包与作用域链
在JavaScript编程中,闭包和作用域链是两个非常重要的概念,它们对于理解代码的执行过程和解决一些特定问题至关重要。本文将深入探讨JavaScript中闭包和作用域链的原理和应用,帮助读者更好地理解这些概念并能够在实际项目中灵活运用。
|
1月前
|
JavaScript 前端开发
javascript闭包的理解(菜菜必看系列!!!)
javascript闭包的理解(菜菜必看系列!!!)