javascript_function(闭包Closure)

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-08-27 00:44:43

javascript_function(闭包Closure)的相关文章

javascript 闭包(closure)

<script type="text/javascript">    //闭包(closure):内层函数可以引用存在于包围它的函数内的变量,即使外层函数的执行已经结束    //注意内层函数引用的外层函数内的变量是外层函数执行结束后的最终值    test(1);    function test(a) { //外层函数        alert(a+' 外层函数开始执行');        setTimeout(function(){//内层函数           

Swift语言精要-闭包(Closure)

闭包(Closure)这个概念如果没学过Swift的人应该也不会陌生. 学过Javascript的朋友应该知道,在Javascript中我们经常会讨论闭包,很多前端工程师的面试题也会问到什么是闭包. 那么,什么是闭包呢? 让我们看下在Javascript中闭包的解释: Closures are functions that have access to variables from another function’s scope. (This is often accomplished by

Swift中使用typealias定义一个闭包closure

在OC中我们定义一个Blocks是这样定义的: typedef void (^ZWProgressHUDCompletionBlock)(); 在Swift中定义一个闭包是这样的: typealias ZWProgressHUDCompletionBlock=()->Void 转载请注明!!!欢迎大家加入交流群:爱疯.爱Coding:209476515 Swift中使用typealias定义一个闭包closure,布布扣,bubuko.com

深入理解JavaScript闭包(closure)

最近在网上查阅了不少Javascript闭包(closure)相关的资料,写的大多是非常的学术和专业.对于初学者来说别说理解闭包了,就连文字叙述都很难看懂.撰写此文的目的就是用最通俗的文字揭开Javascript闭包的真实面目. 一.什么是闭包? “官方”的解释是:闭包是一个拥有许多变量和绑定了这些变量的环境的表达式(通常是一个函数),因而这些变量也是该表达式的一部分.相信很少有人能直接看懂这句话,因为他描述的太学术.其实这句话通俗的来说就是:JavaScript中所有的function都是一个

闭包(closure)与协程共用时要注意的事情

闭包是一种可以让你用非常舒服的方式来编程的小技巧,Go也支持闭包.如果从来没有接触过闭包,想在一开始就弄懂什么是闭包(closure)是非常困难的,就像递归一样,直到你真正写过.用过它,你才能真正的对它有一个更具体的认识. 闭包就是一个函数,这个函数包含了运行它所需的上下文环境,这个环境可能是几个变量或者也会是其他的(通常就是变量).说闭包是一个函数不正确,更确切地说,闭包是一个打包了其作用域外部的上下文环境的一段运行环境.如果一时间没有理解这段闭包的含义也不要紧,这是一个循序渐进的过程. 那么

php中的匿名函数和闭包(closure)

一:匿名函数 (在php5.3.0 或以上才能使用) php中的匿名函数(Anonymous functions), 也叫闭包函数(closures), 允许指定一个没有名称的函数.最常用的就是回调函数的参数值.(http://php.net/manual/zh/functions.anonymous.php) 匿名函数的定义: $closureFunc = function(){ .... }; eg: 把匿名函数赋值给变量,通过变量来调用 $closureFunc = function($s

Python闭包Closure

Python的闭包和Python的内部函数 1 Python内部函数 def out(x): def inner(y): return "inner use % s" % y return inner(x) print out("jeapedu") 在out里定义了一个inner函数,out的返回值是调用inner(x)的值 2 python的闭包Closure def closure(x): def inner(y): return "closure u

[转] Java内部类之闭包(closure)与回调(callback)

闭包(closure)是一个可调用的对象,它记录了一些信息,这些信息来自于创建它的作用域.通过这个定义,可以看出内部类是面向对象的闭包,因为它 不仅包含外围类对象(创建内部类的作用域)的信息,还自动拥有一个指向此外围类对象的引用,在此作用城内,内部类有权操作所有的成员,包括private 成员. Java最引人争议的问题之一就是,人们认为Java应该包含某种类似指针的机制,以允许回调(callback).通过回调,对象能够携带一些信息,这些信息允许它在稍后的某个时刻调用初始的对象. 稍后将会看到

闭包(Closure)和匿名函数(Anonymous function)/lambda表达式的区别

闭包(Closure)和匿名函数(Anonymous function)/lambda表达式的区别 函数最常见的形式是具名函数(named function): function foo(){ console.log("named function") } foo() 不过也可以将函数视作数据赋值给变量,这样的函数可以没有名字: nameless = function(){ console.log("anonymouse function") } nameless(