javascript 闭包

简介:

目录

定义

引子

闭包的用途

一个常见错误

 

 

 


定义

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

引子

首先看一个例子

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 数组中对应的字符串。

 



本文转自 randy_shandong 51CTO博客,原文链接:http://blog.51cto.com/dba10g/1357820,如需转载请自行联系原作者

相关文章
|
5天前
|
自然语言处理 JavaScript 前端开发
JavaScript中闭包:概念、用途与潜在问题
【4月更文挑战第22天】JavaScript中的闭包是函数及其相关词法环境的组合,允许访问外部作用域,常用于数据封装、回调函数和装饰器。然而,不恰当使用可能导致内存泄漏和性能下降。为避免问题,需及时解除引用,减少不必要的闭包,以及优化闭包使用。理解并慎用闭包是关键。
|
5天前
|
JavaScript
闭包(js的问题)
闭包(js的问题)
13 0
|
5天前
|
JavaScript 前端开发
解释JavaScript闭包的工作原理,并举例说明其在游戏开发中的应用。
JavaScript闭包允许内部函数访问并保持对外部函数变量的引用,即使外部函数执行结束。当函数返回内部函数时,形成闭包,继承父函数作用域链。在游戏开发中,闭包用于创建具有独立状态和行为的角色实例。例如,`createCharacter`函数创建角色并返回包含属性和方法的对象,内部函数如`getHealth`、`setHealth`和`attack`通过闭包访问并操作角色的变量。这种方式确保了每个角色的状态在不同实例间独立,是实现游戏逻辑的强大工具。
14 2
|
5天前
|
存储 缓存 JavaScript
|
3天前
|
JavaScript 前端开发
JavaScript 闭包:让你更深入了解函数和作用域
JavaScript 闭包:让你更深入了解函数和作用域
|
3天前
|
自然语言处理 JavaScript 前端开发
JavaScript闭包基础
JavaScript闭包基础
|
5天前
|
缓存 自然语言处理 JavaScript
JavaScript内存泄漏导致应用性能下降,常见于闭包使用不当
【5月更文挑战第14天】JavaScript内存泄漏导致应用性能下降,常见于闭包使用不当。闭包能记住并访问词法作用域,若函数返回后,其引用的对象未被释放,就会引发泄漏。例如,`createLeakyFunction`创建的闭包保留了对大型对象`someLargeObject`的引用,即使函数执行完毕,对象也无法被垃圾回收。避免泄漏的方法包括及时解除引用、清除事件监听器、使用WeakMap和WeakSet以及定期清理缓存。使用性能分析工具可检测和修复内存泄漏问题。
14 3
|
5天前
|
JavaScript 前端开发
JavaScript闭包允许内部函数访问并保留外部函数的变量,即使外部函数执行结束
【5月更文挑战第13天】JavaScript闭包允许内部函数访问并保留外部函数的变量,即使外部函数执行结束。在游戏开发中,闭包常用于创建独立状态的角色实例。例如,`createCharacter`函数生成角色,内部函数(如`getHealth`、`setHealth`)形成闭包,保存角色的属性(如生命值)。这样,每个角色实例都有自己的变量副本,不互相影响,从而实现角色系统的独立性。
21 0
|
5天前
|
自然语言处理 JavaScript 前端开发
深入理解JavaScript中的闭包机制
闭包是JavaScript中一个重要且常被误解的概念。本文将深入探讨闭包的本质、工作原理以及在实际开发中的应用。通过详细解析闭包的定义、作用域链、内存管理等方面,读者将对闭包有更清晰的理解,并能够运用闭包解决实际开发中的问题。
|
5天前
|
前端开发 JavaScript
闭包在JavaScript中有许多应用场景
【5月更文挑战第7天】闭包在JavaScript中发挥关键作用,如封装私有变量和函数提升安全性,维护变量生命周期,实现高阶函数,模拟块级作用域,支持回调函数以处理异步操作,以及促进模块化编程,增强代码组织和管理。闭包是理解和掌握JavaScript高级特性的重要一环。
27 7