Javascript闭包!

# javascript闭包(Closure)
> 所谓“闭包”,指的是一个拥有许多变量和绑定了这些变量的环境的表达式(通常是一个函数),因而这些变量也是该表达式的一部分。

上面是官方的解释,但这解释只会让人头晕。要理解闭包,首先理解两点:变量的作用域以及作用域链,这两个在前面都已经介绍过了,并且举了简单了列子,来回顾一下:

var color = "blue";  
    function changeColor(){     
        var anotherColor = "red";  
        function swapColors(){         
            var tempColor = anotherColor;         
            anotherColor = color;         
            color = tempColor; 
            //在 swapColors函数里面可以访问tempColor,anotherColor和color          
        } 
        //在这里可以访问anotherColor和color,但不能访问tempColor 
        swapColors(); 
    }

上面的代码,我们之前用来说明作用域链,也就是变量从里往外找变量,这一简单的概念。但是,现在反过来,如果,我们需要在一个外部执行环境里面,访问内部执行环境的变量怎么办呢?
    
    function f1(){
        var n=999;
        function f2(){
            alert(n); // 999
        }
        //f2();
    }

上面的代码很简单,我们在f1里面去调用f2()当然可以弹出n的值999,但是如果我们在外部全局执行环境里面还要获得n的值,这又该怎么办?这就是刚刚的提问,外部的执行环境要求访问内部执行环境的值。
从外往里访问,这里就可以通过闭包来实现这个效果,把上面的代码稍作修改:

function f1(){
        var n=999; //私有变量
    
        //在函数f1内定义另外的函数作为f1的方法函数
        function f2(){
            alert(n); //引用外层函数f1的临时变量n
        }
        return f2; //返回内部函数
    }
    //调用函数
    var result=f1();
    result(); // 999

__当其中一个这样的内部函数在包含它们的外部函数之外被调用时,就会形成闭包__(示例中,调用函数的时候,result实际调用的是f2函数,也就是说f1的一个内部函数f2在f1之外被调用,这时就创建了一个闭包)。

在看一个简单例子:

function foo() {
        var a = 1;
        function geta() {
            a++;
            return a;
        }
        return geta
    }
    
    myfunc = foo()
    myfunc() // return 2
    myfunc() //return 3

上面只是一个说明闭包的例子,在实际应用中并不常见。下面我们来看看实际应用中会遇到闭包的情况

__情况一__
界面上有一组a标签,分别点击,希望点击不同的a标签弹出不同的结果,代码如下:

<ul>
        <li><a href="#">第0个链接</a></li>
        <li><a href="#">第1个链接</a></li>
        <li><a href="#">第2个链接</a></li>
        <li><a href="#">第3个链接</a></li>
    </ul>    
    
    var as = document.getElementsByTagName("a");
    for(var i=0;i<as.length;i++){
        as[i].onclick = function(){
            alert("你现在单击的是第" + i + "个链接");
        }
    }

上面这段代码的估计大家都遇到过类似的,想要的效果和实际出现的效果不一致,本来想点击每一个,弹出不同的i的值,但是没想到最后弹出的i的值都是4。
这正是由于变量的作用域链影响造成的,因为在用户单击a标签的时候才会调用onclick指向的匿名函数。而调用时,需要得到变量i的值,js解析程序首先会在匿名函数内部查找i,但是没有定义。于是往外找,在外部执行环境中找到变量i,但是这个i已经被循环到4了,因为for循环在页面初始化的时候已经被执行了。所以匿名函数里面的i实际上取得的是外部作用域链中i的值。想要改正这个错误,其实就是把i的值传入到匿名函数值就行了。但是匿名函数又不能传值。这个时候,我们就可以用闭包。
_先创建这样一个函数:_

function closureTest(num){
        return function(){
            alert("你现在单击的是第" + num + "个链接")
        }
    }

这个函数直接返回了一个匿名函数,之前讲过闭包,所以,当返回的这个函数在外部被接收的时候,__外层函数closureTest里的变量num,就会被保存起来__,所以,将之前循环的代码修改一下:

for(var i=0;i<as.length;i++){
        as[i].onclick = closureTest(i);
    }

循环里面onclick调用的函数closureTest,并传入了参数i,而closureTest返回了匿名函数,所以根据闭包的原理,传入的参数i,就被保存在了内存当中,外部访问的就是每次不一样的值了。当然,你也可以直接写成下面这个样子:

for(var i=0;i<as.length;i++){
        as[i].onclick = (function(i){
            return function(){
                alert("你现在单击的是第" + i + "个链接")
            }
        })(i);
    }

__情况二__
__利用闭包巧妙地传递参数__,比如,有这样的场景,点击标签,然后延迟弹出一句话,而而这句话,是用参数传递过去的。先看下面的代码:

<a href="#">点击我</a>
    
    var link = document.getElementById("myLink");
    link.onclick = function(){
        setTimeout(function(){
            alert("你点击了点击我的超链接!");
        },1000);
    }

这段代码,就是点击1秒后,弹出"你点击了点击我的超链接!",当然这里是直接把话写在了function里面,如果这段话是从外面传过来的呢?比如有个方法

function someFunction(words){
        alert(words);
    }
    var link = document.getElementById("myLink");
    link.onclick = function(){
        setTimeout(someFunction,1000);
    };

这里的话,那么问题就来了。怎么往someFunction传递参数呢?这个也可以直接用闭包

function someFunction(words){
        return function(){
            alert(words);
        };
    }
    var link = document.getElementById("myLink");
    link.onclick = function(){
        var saySomething = someFunction("你点击了点击我的超链接!");
        setTimeout(saySomething,1000);
    };

__在动态执行环境中,数据实时地发生变化,为了保持这些非持久型变量的值,我们用闭包这种载体来存储这些动态数据。这就是闭包的作用。也就说遇到需要存储动态变化的数据或将被回收的数据时,我们可以通过外面再包裹一层函数形成闭包来解决。__

时间: 2024-11-03 22:51:36

Javascript闭包!的相关文章

深入理解javascript闭包

闭包(closure)是Javascript语言的一个难点,也是它的特色,很多高级应用都要依靠闭包实现. 一.变量的作用域 要理解闭包,首先必须理解Javascript特殊的变量作用域. 变量的作用域无非就是两种:全局变量和局部变量. Javascript语言的特殊之处,就在于函数内部可以直接读取全局变量. Js代码 var n=999; function f1(){ alert(n); } f1(); // 999 另一方面,在函数外部自然无法读取函数内的局部变量. Js代码 function

JavaScript 闭包

1.词法作用域: 简单地说子集能访问父级的变量, 说人话就是变量拿来就用不用传入 2.函数局部变量: 在函数体中以var 声明变量的为局部变量 + 函数传入的参数, 直接写变量名声明的变量是全局变量 3.局部变量生存期: 局部变量在函数函数的执行期间可用,  一旦执行过后,局部变量将不再可用 4.延长局部变量生存期: 现在问题来了,我想要延长局部变量的生存期,怎么办.(因为调用函数不仅仅是为了return, 有时候还需要保存函数中的状态, 或者实现类等等) 5.使用全局变量不好吗: 不好.有时函

javascript—闭包

javascript 闭包就是在另一个作用域中保存了一份它从上一级函数或作用域取得的变量(键值对0), 而这些键值对是不会随上一级函数的执行完成而销毁. function a(){ var i=0; function b(){ alert(++i); } return b; } var c=a(); c(); 在执行完var c=a()后,变量c实际上是指向了函数b,b中用到了变量i, 再执行c()后就会弹出一个窗口显示i的值(第一次为1).这段代码其实就创建了一个闭包. 为什么?因为函数a外的

JavaScript 闭包究竟是什么

JavaScript 闭包究竟是什么 1.简单的例子 首先从一个经典错误谈起,页面上有若干个div, 我们想给它们绑定一个onclick方法,于是有了下面的代码 <div id="divTest"> <span>0</span> <span>1</span> <span>2</span> <span>3</span> </div> <div id="d

全面理解Javascript闭包和闭包的几种写法及用途

一.什么是闭包和闭包的几种写法和用法                                                       1.什么是闭包 闭包,官方对闭包的解释是:一个拥有许多变量和绑定了这些变量的环境的表达式(通常是一个函数),因而这些变量也是该表达式的一部分.闭包的特点: 1. 作为一个函数变量的一个引用,当函数返回时,其处于激活状态. 2. 一个闭包就是当一个函数返回时,一个没有释放资源的栈区. 简单的说,Javascript允许使用内部函数---即函数定义和函数表

javascript闭包的简单理解

奖Javascript闭包前,先给大家讲个小故事 故事背景:刘备和曹操煮酒的时间段,故事可能有点小差异,刘备为曹操手下,关羽为间谍. 咳咳 曹操很想知道手下新来的刘备整天鼓捣啥: 曹操就问刘备,玄德你整天鼓捣啥: 刘备想,我靠这怎么能告诉你,劳资岂不是找死,打个马虎眼就过去了,曹操什么也没问出来: 刘备回到家里,跟关羽说,二弟,哼,曹操那个傻吊,哼! 关羽晚上也回到家里,跟老婆说,哼,大哥那个傻吊,哼!我这就去告诉丞相: 曹操也不能罢休,就去问关羽,关羽正要去告密,就一五一十  ¥%…@?%%:

javascript 闭包学习

闭包(closure)是Javascript语言的一个难点,也是它的特色,很多高级应用都要依靠闭包实现. 一.变量的作用域 要理解闭包,首先必须理解Javascript特殊的变量作用域. 变量的作用域无非就是两种:全局变量和局部变量. Javascript语言的特殊之处,就在于函数内部可以直接读取全局变量. Js代码 var n=999; function f1(){ alert(n); } f1(); // 999 另一方面,在函数外部自然无法读取函数内的局部变量. Js代码 function

javascript闭包详解(内容为转载的,觉得不错就分享一下)

一.变量的作用域 要理解闭包,首先必须理解Javascript特殊的变量作用域. 变量的作用域无非就是两种:全局变量和局部变量. Javascript语言的特殊之处,就在于函数内部可以直接读取全局变量. Js代码 var n=999; function f1(){ alert(n); } f1(); // 999 另一方面,在函数外部自然无法读取函数内的局部变量. Js代码 function f1(){ var n=999; } alert(n); // error 这里有一个地方需要注意,函数

转 全面理解Javascript闭包和闭包的几种写法及用途

转自:http://www.cnblogs.com/yunfeifei/p/4019504.html 好久没有写博客了,过了一个十一长假都变懒了,今天总算是恢复状态了.好了,进入正题,今天来说一说javascript里面的闭包吧!本篇博客主要讲一些实用的东西,主要将闭包的写法.用法和用途. 一.什么是闭包和闭包的几种写法和用法                                                       1.什么是闭包 闭包,官方对闭包的解释是:一个拥有许多变量和绑

javascript闭包的理解-韩烨

javascript闭包是javascript的难点,很多人对js闭包不是很理解,我对js闭包一开始也是云里雾里,我刚刚进兴安得力的时候,做的转正试题中就有一个对闭包理解的题目.如何理解javascript的闭包呢?下面我们一起来学习一下: 闭包的含义和理解 通俗地讲,JavaScript 中每个的函数都是一个闭包,但通常意义上嵌套的函数更能够体现出闭包的特性,请看下面这个例子: var generateClosure = function() { var count = 0; var get