Javascript 中闭包(Closure)的探索(二)-私有变量和函数

简介:

利用匿名函数形成闭包可以在javascript中实现面向对象语言中的访问权限控制。即在javascript中也能实现私有变量。

参考网址:http://www.crockford.com/javascript/private.html

 

1.构造私有变量和公有变量

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
<html xmlns= "http://www.w3.org/1999/xhtml"  >
<head>
     <title>JsClosure2</title>
     <script type= "text/javascript" >
         function  ClassFunc() {
             this .publicMem = "public" ;
             var  privateMem = "private" ;
         }
 
         function  closureTestClick() {
             var  test = new  ClassFunc();
             alert(test.publicMem);
             alert(test.privateMem);
         }
     </script>
</head>
<body>
<input type= "button"  value= "closureTest"  onclick= "closureTestClick()"  />
</body>
</html>

结果:alert(test.publicMem);可以正常显示,alert(test.privateMem);显示“undefined”。

结果分析:通过var定义的私有变量外界无法访问,如果要外界可以访问,需要构造get,set方法。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
<script type= "text/javascript" >
         function  ClassFunc() {
             this .publicMem = "public" ;
             var  privateMem = "private" ;
             this .getprivateMem = function () {
                 return  privateMem;
             }
             this .setprivateMem = function (val) {
                 privateMem = val;
             }
         }
 
         function  closureTestClick() {
             var  test = new  ClassFunc();
             alert(test.getprivateMem());
             test.setprivateMem( "private changed!" );
             alert(test.getprivateMem());
         }
     </script>

结果:如预期的一样显示“private”和“private changed!”。

 

2.私有函数

与私有变量的定义类似,不是通过this来定义的函数都是私有函数。

私有函数外部无法调用,但是可以通过内部的公有函数来调用。

测试代码如下:

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
33
<script type= "text/javascript" >
         function  ClassFunc() {
             this .publicMem = "public" ;
             var  privateMem = "private" ;
             this .getprivateMem = function () {
                 return  privateMem;
             }
             this .setprivateMem = function (val) {
                 privateMem = val;
             }
 
             function  privateFunc() {
                 privateMem = "private changed!" ;
                 // 此处的赋值并没有如预期的那样给test.publicMem成员赋值
                 this .publicMem = "public changed!" ;
             }
 
             this .callprivateFunc = function () {
                 privateFunc();
             }
         }
 
         function  closureTestClick() {
             var  test = new  ClassFunc();
             // 变更前
             alert( "privateMem=" +test.getprivateMem());
             alert( "publicMem="  + test.publicMem);
             test.callprivateFunc();
             // 变更后
             alert( "privateMem="  + test.getprivateMem());
             alert( "publicMem="  + test.publicMem);
         }
     </script>

变更后的结果privateMem如预期一样,而publicMem仍然是“public”,并没有改变。

这是因为函数privateFunc() 中this.publicMem 的this已经不是指向test这个js对象了。

关于this的指向为什么会变,参见我的第三篇文章,介绍javascript的scope的。

本例中为了能够修改testpublicMem属性,有两个方法:

其一,也是常用的,直接在外部修改publicMem,因为publicMem是公有变量。

1
test.publicMem = "public changed!" ;

其二,在函数callprivateFunc 和privateFunc中增加一个参数,显示的传入test对象。

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
33
<script type= "text/javascript" >
         function  ClassFunc() {
             this .publicMem = "public" ;
             var  privateMem = "private" ;
             this .getprivateMem = function () {
                 return  privateMem;
             }
             this .setprivateMem = function (val) {
                 privateMem = val;
             }
 
             function  privateFunc(obj) {
                 privateMem = "private changed!" ;
                 // 直接给obj.publicMem赋值
                 obj.publicMem = "public changed!" ;
             }
 
             this .callprivateFunc = function (obj) {
                 privateFunc(obj);
             }
         }
 
         function  closureTestClick() {
             var  test = new  ClassFunc();
             // 变更前
             alert( "privateMem=" +test.getprivateMem());
             alert( "publicMem="  + test.publicMem);
             test.callprivateFunc(test);
             // 变更后
             alert( "privateMem="  + test.getprivateMem());
             alert( "publicMem="  + test.publicMem);
         }
     </script>
标签:  javascript学习



本文转自wang_yb博客园博客,原文链接:http://www.cnblogs.com/wang_yb/archive/2010/05/06/1729070.html,如需转载请自行联系原作者
目录
相关文章
|
9月前
|
机器学习/深度学习 JavaScript 前端开发
JS进阶教程:递归函数原理与篇例解析
通过对这些代码示例的学习,我们已经了解了递归的原理以及递归在JS中的应用方法。递归虽然有着理论升华,但弄清它的核心思想并不难。举个随手可见的例子,火影鸣人做的影分身,你看到的都是同一个鸣人,但他们的行为却能在全局产生影响,这不就是递归吗?雾里看花,透过其间你或许已经深入了递归的魅力之中。
353 19
|
前端开发 JavaScript Java
JavaScript闭包深入剖析:性能剖析与优化技巧
JavaScript 闭包是强大而灵活的特性,广泛应用于数据封装、函数柯里化和事件处理等场景。闭包通过保存外部作用域的变量,实现了私有变量和方法的创建,提升了代码的安全性和可维护性。然而,闭包也可能带来性能问题,如内存泄漏和执行效率下降。为优化闭包性能,建议采取以下策略:及时解除对不再使用的闭包变量的引用,减少闭包的创建次数,使用 WeakMap 管理弱引用,以及优化闭包结构以减少作用域链查找的开销。在实际开发中,无论是 Web 前端还是 Node.js 后端,这些优化措施都能显著提升程序的性能和稳定性。
307 70
|
11月前
|
存储 JavaScript 前端开发
|
自然语言处理 JavaScript 前端开发
当面试官再问我JS闭包时,我能答出来的都在这里了。
闭包(Closure)是前端面试中的高频考点,广泛应用于函数式编程中。它不仅指函数内部定义的函数,还涉及内存管理、作用域链和垃圾回收机制。闭包可以让函数访问其外部作用域的变量,但也可能引发内存泄漏等问题。通过合理使用闭包,可以实现模块化、高阶函数和回调函数等应用场景。然而,滥用闭包可能导致代码复杂度增加、调试困难以及潜在的性能问题。为了避免这些问题,开发时应谨慎处理闭包,避免不必要的嵌套,并及时清理不再使用的变量和监听器。
508 16
当面试官再问我JS闭包时,我能答出来的都在这里了。
|
11月前
|
JavaScript
JS实现多条件搜索函数
JS封装的多条件搜索
|
JavaScript 前端开发
JavaWeb JavaScript ③ JS的流程控制和函数
通过本文的详细介绍,您可以深入理解JavaScript的流程控制和函数的使用,进而编写出高效、可维护的代码。
273 32
|
JavaScript 前端开发
js 闭包的优点和缺点
【10月更文挑战第27天】JavaScript闭包是一把双刃剑,在合理使用的情况下,它可以带来很多好处,如实现数据封装、记忆功能和模块化等;但如果不注意其缺点,如内存泄漏、变量共享和性能开销等问题,可能会导致代码出现难以调试的错误和性能问题。因此,在使用闭包时,需要谨慎权衡其优缺点,根据具体的应用场景合理地运用闭包。
412 58
|
12月前
|
缓存 自然语言处理 JavaScript
JavaScript中闭包详解+举例,闭包的各种实践场景:高级技巧与实用指南
闭包是JavaScript中不可或缺的部分,它不仅可以增强代码的可维护性,还能在模块化、回调处理等场景中发挥巨大作用。然而,闭包的强大也意味着需要谨慎使用,避免潜在的性能问题和内存泄漏。通过对闭包原理的深入理解以及在实际项目中的灵活应用,你将能够更加高效地编写出简洁且功能强大的代码。 只有锻炼思维才能可持续地解决问题,只有思维才是真正值得学习和分享的核心要素。如果这篇博客能给您带来一点帮助,麻烦您点个赞支持一下,还可以收藏起来以备不时之需,有疑问和错误欢迎在评论区指出~
|
12月前
|
JavaScript 前端开发 Java
详解js柯里化原理及用法,探究柯里化在Redux Selector 的场景模拟、构建复杂的数据流管道、优化深度嵌套函数中的精妙应用
柯里化是一种强大的函数式编程技术,它通过将函数分解为单参数形式,实现了灵活性与可复用性的统一。无论是参数复用、延迟执行,还是函数组合,柯里化都为现代编程提供了极大的便利。 从 Redux 的选择器优化到复杂的数据流处理,再到深度嵌套的函数优化,柯里化在实际开发中展现出了非凡的价值。如果你希望编写更简洁、更优雅的代码,柯里化无疑是一个值得深入学习和实践的工具。从简单的实现到复杂的应用,希望这篇博客能为你揭开柯里化的奥秘,助力你的开发之旅! 只有锻炼思维才能可持续地解决问题,只有思维才是真正值得学习和分享的核心要素。如果这篇博客能给您带来一点帮助,麻烦您点个赞支持一
|
缓存 JavaScript 前端开发
js 闭包
【10月更文挑战第27天】JavaScript闭包是一种强大的特性,它可以用于实现数据隐藏、记忆和缓存等功能,但在使用时也需要注意内存泄漏和变量共享等问题,以确保代码的质量和性能。
303 7

热门文章

最新文章