`
dreamoftch
  • 浏览: 485185 次
  • 性别: Icon_minigender_1
  • 来自: 上海
社区版块
存档分类
最新评论

javascript 闭包的简单例子解释

阅读更多


http://hzjavaeyer.group.iteye.com/group/wiki/1317-javascript-object-oriented-technology-6

http://www.iteye.com/wiki/javascript-core

作用域、闭包、模拟私有属性

先来简单说一下变量作用域,这些东西我们都很熟悉了,所以也不详细介绍。
Js代码  收藏代码

    var sco = "global";  //全局变量 
    function t() {  
        var sco = "local";  //函数内部的局部变量 
        alert(sco);         //local 优先调用局部变量 
    } 
    t();             //local 
    alert(sco);       //global  不能使用函数内的局部变量 



注意一点,在javascript中没有块级别的作用域,也就是说在java或c/c++中我们可以
用"{}"来包围一个块,从而在其中定义块内的局部变量,在"{}"块外部,这些变量不再起作用,
同时,也可以在for循环等控制语句中定义局部的变量,但在javascript中没有此项特性:
Js代码  收藏代码

    function f(props) { 
        for(var i=0; i<10; i++) {} 
        alert(i);         //10  虽然i定义在for循环的控制语句中,但在函数 
                          //的其他位置仍旧可以访问该变量. 
        if(props == "local") { 
            var sco = "local"; 
        alert(sco);  
        } 
        alert(sco);       //同样,函数仍可引用if语句内定义的变量 
    } 
    f("local");      //10  local   local 



在函数内部定义局部变量时要格外小心:
Js代码  收藏代码

    var sco = "global"; 
    function print1() { 
        alert(sco);   //global 
    } 
    function print2() { 
        var sco = "local"; 
        alert(sco);   //local 
    } 
    function print3() { 
        alert(sco);   //undefined 
        var sco = "local";  
        alert(sco);   local 
    } 
     
    print1();  //global 
    print2();  //local 
    print3();  //undefined  local 

前面两个函数都很容易理解,关键是第三个:第一个alert语句并没有把全局变量"global"显示出来,
而是undefined,这是因为在print3函数中,我们定义了sco局部变量(不管位置在何处),那么全局的
sco属性在函数内部将不起作用,所以第一个alert中sco其实是局部sco变量,相当于:
Js代码  收藏代码

    function print3() { 
        var sco; 
        alert(sco); 
        sco = "local"; 
        alert(sco); 
    } 

从这个例子我们得出,在函数内部定义局部变量时,最好是在开头就把所需的变量定义好,以免出错。

函数的作用域在定义函数的时候已经确定了,例如:
Js代码  收藏代码

    var scope = "global"   //定义全局变量 
    function print() { 
        alert(scope); 
    } 
    function change() { 
        var scope = "local";  //定义局部变量 
        print();              //虽然是在change函数的作用域内调用print函数, 
                              //但是print函数执行时仍旧按照它定义时的作用域起作用 
    } 
    change();    //golbal 



闭包
闭包是拥有变量、代码和作用域的表达式.在javascript中,函数就是变量、代码和函数的作用域的组合体,因此所有
的函数都是闭包(JavaScript functions are a combination of code to be executed and the scope in which to
execute them. This combination of code and scope is known as a closure in the computer science literature.
All JavaScript functions are closures).好像挺简单.但是闭包到底有什么作用呢?看一个例子。
我们想写一个方法,每次都得到一个整数,这个整数是每次加1的,没有思索,马上下笔:
Js代码  收藏代码

    var i = 0; 
    function getNext() { 
        i++; 
        return i; 
    } 
    alert(getNext()); //1 
    alert(getNext()); //2 
    alert(getNext()); //3 



一直用getNext函数得到下一个整数,而后不小心或者故意的将全局变量i的值设为0,然后再次调用getNext,
你会发现又从1开始了........这时你会想到,要是把i设置成一个私有变量该多好,这样只有在方法内部才
可能改变它,在函数之外就没有办法修改了.下面的代码就是按照这个要求来做得,后面我们详细讨论。
为了解释方便,我们就把下面的代码称为demo1.
Js代码  收藏代码

    function temp() { 
        var i = 0; 
        function b() { 
            return ++i; 
        } 
        return b; 
    } 
    var getNext = temp(); 
    alert(getNext());    //1 
    alert(getNext());    //2 
    alert(getNext());    //3 
    alert(getNext());    //4 



因为我们平时所说的javascript绝大多数都是指的在客户端(浏览器)下,所以这里也不例外。
在javascript解释器启动时,会首先创建一个全局的对象(global object),也就是"window"所引用的对象.
然后我们定义的所有全局属性和方法等都会成为这个对象的属性.
不同的函数和变量的作用域是不同的,因而构成了一个作用域链(scope chain).很显然,在javascript解释器启动时,
这个作用域链只有一个对象:window(Window Object,即global object).
在demo1中,temp函数是一个全局函数,因此temp()函数的作用域(scopr)对应的作用域链就是js解释器启动时的作用域链,只有一个window对象。
当temp执行时,首先创建一个call对象(活动对象),然后把这个call对象添加到temp函数对应的作用域链的最前头,这是,temp()函数
对应的作用域链就包含了两个对象:window对象和temp函数对应的call object(活动对象).然后呢,因为我们在temp函数里定义了变量i,
定义了函数b(),这些都会成为call object的属性。当然,在这之前会首先给call object对象添加arguments属性,保存了temp()函数执行时
传递过来的参数。此时,整个的作用域链如下图所示:


同理可以得出函数b()执行时的整个作用域链:


注意在b()的作用域链中,b()函数对应的call object只有一个arguemnts属性,并没有i属性,这是因为在b()的定义中,并没有用var关键字来
声明i属性,只有用var 关键字声明的属性才会添加到对应的call object上.
在函数执行时,首先查找对应的call object有没有需要的属性,如果没有,再往上一级查找,直到找到为止,如果找不到,那就是undefined了.

这样我们再来看demo1的执行情况。我们用getNext引用了temp函数,而temp函数返回了函数b,这样getNext函数其实就是b函数的引用。
执行一次getNext,就执行一次b()函数。因为函数b()的作用域依赖于函数temp,因此temp函数在内存中会一直存在。函数b执行时,首先查找
i,在b对应的call object中没有,于是往上一级找,在temp函数对应的call object中找到了,于是将其值加1,然后返回这个值。
这样,只要getNext函数有效,那么b()函数就一直有效,同时,b()函数依赖的temp函数也不会消失,变量i也不会消失,而且这个变量在temp函数
外部根本就访问不到,只能在temp()函数内部访问(b当然可以了).

来看一个利用闭包来模拟私有属性的例子:
Js代码  收藏代码

    function Person(name, age) {   
        this.getName = function() { return name; };   
        this.setName = function(newName) { name = newName; };   
        this.getAge = function() { return age; };   
        this.setAge = function(newAge) { age = newAge; };   
    }   
       
    var p1 = new Person("sdcyst",3);   
    alert(p1.getName());  //sdcyst   
    alert(p1.name);       //undefined   因为Person('类')没有name属性   
    p1.name = "mypara"    //显示的给p1添加name属性   
    alert(p1.getName());  //sdcyst     但是并不会改变getName方法的返回值   
    alert(p1.name);       //mypara     显示出p1对象的name属性   
    p1.setName("sss");    //改变私有的"name"属性 
    alert(p1.getName());  //sss   
    alert(p1.name);       //仍旧为mypara   



定义了一个Person类,有两个私有属性name,age,分别定义对应的get/set方法。
虽然可以显示的设置p1的name、age属性,但是这种显示的设置,并不会改变我们
最初设计时模拟出来的"name/age"私有属性。


分享到:
评论

相关推荐

    javascript 闭包实例下载

    javascript 闭包实例下载

    JavaScript闭包

    Javascript中有几个非常重要的语言特性——对象、原型继承、闭包。其中闭包 对于那些使用传统静态...本文将以例子入手来介绍Javascript闭包的语言特性,并结合一点 ECMAScript语言规范来使读者可以更深入的理解闭包。

    javascript闭包详解中文word版

    本文将以例子入手来介绍Javascript闭包的语言特性,并结合一点 ECMAscript语言规范来使读者可以更深入的理解闭包。闭包是Closure, 资源太大,传百度网盘了,链接在附件中,有需要的同学自取。

    javascript闭包的高级使用方法实例

    介绍了javascript闭包的高级使用方法实例,有需要的朋友可以参考一下

    javascript实现的闭包简单实例

    主要介绍了javascript实现的闭包简单实现方法,涉及javascript闭包的原理与实现技巧,具有一定参考借鉴价值,需要的朋友可以参考下

    Javascript闭包用法实例分析

    本文实例分析了Javascript闭包的概念及用法。分享给大家供大家参考。具体如下: 提到闭包,想必大家都早有耳闻,下面说下我的简单理解。 说实话平时工作中实际手动写闭包的场景并不多,但是项目中用到的第三方框架和...

    javascript 闭包详解及简单实例应用

    主要介绍了javascript 闭包详解及应用的相关资料,需要的朋友可以参考下

    Javascript闭包实例详解

    闭包就是函数的局部变量集合,只是这些局部变量在函数返回后会继续存在,本文通过代码实例给大家介绍javascript闭包,对javascipt闭包相关知识感兴趣的朋友一起学习吧

    javaScript闭包的理解

    网上的例子很多,讲解的和理解起来的简易程度不禁相同,整理的例子和吸收的帖子供大家参考

    JavaScript闭包和范围实例详解

    主要介绍了JavaScript闭包和范围,结合实例形式详细分析了javascript闭包的使用及变量的作用范围,需要的朋友可以参考下

    最简单的例子让你轻松理解JavaScript闭包

    在理解闭包之前,为了让大家更好的理解闭包,我举一个形象的例子。在JS里有着作用域的概念,那么作用域是什么呢?有什么用呢?下面这个例子能形象的体现 比如每一座城市,城市里乡镇啊有公园,超市等,这些建筑设施...

    Javascript 闭包详解及实例代码

    Javascript 闭包 闭包,是 Javascript 比较...其实,闭包无处不在,比如:jQuery、zepto的核心代码都包含在一个大的闭包中,所以下面我先写一个最简单最原始的闭包,以便让你在大脑里产生闭包的画面: function A(){

    JavaScript闭包和回调详解

    本文主要讲解了JavaScript闭包和回调,闭包的概念和特性,结合实例分析了使用步骤与方法

    javascript闭包的理解和实例

    下面是一个简单的使用全局变量的闭包实例: 代码如下: var sWord=”Hello,Welcome to web前端开发工程师的博客,请多多指教。” function disWord(){ alert&#40;sWord&#41;; } disWord(); 解析:脚本载入到内存的...

    JavaScript闭包与作用域链实例分析

    本文实例讲述了JavaScript闭包与作用域链。分享给大家供大家参考,具体如下: 闭包定义 闭包指的是有权访问另一个函数作用域中的变量的函数。创建闭包的常见方式,就是在一个函数A内部创建另一个函数B,那么函数B...

    JavaScript闭包实例详解

    主要介绍JavaScript闭包知识,包括闭包的基本概念,闭包的用途等相关知识,介绍的非常详细,具有参考借鉴价值,感兴趣的朋友一起看看吧

    javascript闭包(Closure)用法实例简析

    本文实例讲述了javascript闭包(Closure)用法。分享给大家供大家参考,具体如下: closure被翻译成“闭包”,感觉这东西被包装的太学术化。下面参考书本和网上资源简单探讨一下(理解不当之处务请留意)。 1、什么是...

Global site tag (gtag.js) - Google Analytics