谈一谈javascript闭包

 更新时间:2016年01月28日 11:27:37   投稿:lijiao  
这篇文章主要介绍了javascript闭包,闭包(closure)是Javascript语言的一个难点,也是它的特色,很多高级应用都要依靠闭包实现,感兴趣的小伙伴们可以参考一下

下面就是我的学习笔记,对于Javascript初学者应该是很有用的。

一、变量的作用域
要理解闭包,首先必须理解Javascript特殊的变量作用域。
变量的作用域无非就是两种:全局变量和局部变量。
Javascript语言的特殊之处,就在于函数内部可以直接读取全局变量。

 var n=999; 
  function f1(){ 
    alert(n); 
  } 
  f1(); // 999 

另一方面,在函数外部自然无法读取函数内的局部变量。

  function f1(){ 
    var n=999; 
  } 
  alert(n); // error 

这里有一个地方需要注意,函数内部声明变量的时候,一定要使用var命令。如果不用的话,你实际上声明了一个全局变量!

 function f1(){ 
    n=999; 
  } 
  f1(); 
  alert(n); // 999 

二、如何从外部读取局部变量?
出于种种原因,我们有时候需要得到函数内的局部变量。但是,前面已经说过了,正常情况下,这是办不到的,只有通过变通方法才能实现。
那就是在函数的内部,再定义一个函数。

  function f1(){ 
    var n=999; 
    function f2(){ 
      alert(n); // 999 
    } 
  } 

在上面的代码中,函数f2就被包括在函数f1内部,这时f1内部的所有局部变量,对f2都是可见的。但是反过来就不行,f2内部的局部变量,对f1 就是不可见的。这就是Javascript语言特有的”链式作用域”结构(chain scope),子对象会一级一级地向上寻找所有父对象的变量。所以,父对象的所有变量,对子对象都是可见的,反之则不成立。
既然f2可以读取f1中的局部变量,那么只要把f2作为返回值,我们不就可以在f1外部读取它的内部变量了吗!

  function f1(){ 
    var n=999; 
    function f2(){ 
      alert(n); 
    } 
    return f2; 
  } 
  var result=f1(); 
  result(); // 999 

三、闭包的概念
上一节代码中的f2函数,就是闭包。
各种专业文献上的”闭包”(closure)定义非常抽象,很难看懂。我的理解是,闭包就是能够读取其他函数内部变量的函数。
由于在Javascript语言中,只有函数内部的子函数才能读取局部变量,因此可以把闭包简单理解成”定义在一个函数内部的函数”。
所以,在本质上,闭包就是将函数内部和函数外部连接起来的一座桥梁。

四、闭包的用途
闭包可以用在许多地方。它的最大用处有两个,一个是前面提到的可以读取函数内部的变量,另一个就是让这些变量的值始终保持在内存中。
怎么来理解这句话呢?请看下面的代码。

  function f1(){ 
    var n=999; 
    nAdd=function(){n+=1} 
    function f2(){ 
      alert(n); 
    } 
    return f2; 
  } 
  var result=f1(); 
  result(); // 999 
  nAdd(); 
  result(); // 1000 

在这段代码中,result实际上就是闭包f2函数。它一共运行了两次,第一次的值是999,第二次的值是1000。这证明了,函数f1中的局部变量n一直保存在内存中,并没有在f1调用后被自动清除。
为什么会这样呢?原因就在于f1是f2的父函数,而f2被赋给了一个全局变量,这导致f2始终在内存中,而f2的存在依赖于f1,因此f1也始终在内存中,不会在调用结束后,被垃圾回收机制(garbage collection)回收。
这段代码中另一个值得注意的地方,就是”nAdd=function(){n+=1}”这一行,首先在nAdd前面没有使用var关键字,因此 nAdd是一个全局变量,而不是局部变量。其次,nAdd的值是一个匿名函数(anonymous function),而这个匿名函数本身也是一个闭包,所以nAdd相当于是一个setter,可以在函数外部对函数内部的局部变量进行操作。

五、使用闭包的注意点
1)由于闭包会使得函数中的变量都被保存在内存中,内存消耗很大,所以不能滥用闭包,否则会造成网页的性能问题,在IE中可能导致内存泄露。解决方法是,在退出函数之前,将不使用的局部变量全部删除。
2)闭包会在父函数外部,改变父函数内部变量的值。所以,如果你把父函数当作对象(object)使用,把闭包当作它的公用方法(Public Method),把内部变量当作它的私有属性(private value),这时一定要小心,不要随便改变父函数内部变量的值。

六、思考题
如果你能理解下面两段代码的运行结果,应该就算理解闭包的运行机制了。
代码片段一:

 var name = “The Window”; 
  var object = { 
    name : “My Object”, 
    getNameFunc : function(){ 
      return function(){ 
        return this.name; 
      }; 
    } 
  }; 
  alert(object.getNameFunc()());

代码片段二:

 var name = “The Window”; 
  var object = { 
    name : “My Object”, 
    getNameFunc : function(){ 
      var that = this; 
      return function(){ 
        return that.name; 
      }; 
    } 
  }; 
  alert(object.getNameFunc()()); 

还有一个是this在javascript中是如何工作的?

var fullname = ‘John Doe'; 
var obj = { 
fullname: ‘Colin Ihrig', 
prop: { 
fullname: ‘Aurelio De Rosa', 
getFullname: function() { 
return this.fullname; 
} 
} 
};

console.log(obj.prop.getFullname());

var test = obj.prop.getFullname;

console.log(test());

回答
答案是Aurelio De Rosa和John Doe。原因是,在一个函数中,this的行为,取决于JavaScript函数的调用方式和定义方式,而不仅仅是看它如何被定义的。
在第一个 console.log()调用中,getFullname() 被调用作为obj.prop对象的函数。所以,上下文指的是后者,函数返回该对象的fullname。与此相反,当getFullname()被分配到test变量时,上下文指的是全局对象(window)。这是因为test是被隐式设置为全局对象的属性。出于这个原因,该函数返回window的fullname,即定义在第一行的那个值。

以上就是本文的全部内容,希望对大家再一次理解javascript闭包有所帮助。

相关文章

  • JavaScript定时器原理详解

    JavaScript定时器原理详解

    这篇文章主要介绍了JavaScript定时器原理,setTimeout()方法用于设置一个定时器,该定时器在定时器到期后执行调用函数,文章围绕JavaScript定时器相关资料展开以下内容,需要的朋友可以参考一下
    2021-12-12
  • 基于javascript的无缝滚动动画1

    基于javascript的无缝滚动动画1

    这篇文章主要介绍了基于javascript的无缝滚动动画实现,文章通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-08-08
  • js防刷新的倒计时代码 js倒计时代码

    js防刷新的倒计时代码 js倒计时代码

    这篇文章主要为大家详细介绍了js防刷新的倒计时代码,js倒计时的实现代码,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-09-09
  • 在WordPress中加入Google搜索功能的简单步骤讲解

    在WordPress中加入Google搜索功能的简单步骤讲解

    这篇文章主要介绍了在WordPress中加入Google搜索的简单步骤讲解,谷歌搜索很有用,但同时也要注意在国内使用cse的连通性,需要的朋友可以参考下
    2016-01-01
  • JS实现短信验证码一键登录功能

    JS实现短信验证码一键登录功能

    短信验证码一键登录是一种方便快捷的登录方式,本文介绍了其原理并给出了一个简单的JavaScript示例,感兴趣的朋友跟随小编一起看看吧
    2024-05-05
  • JS实现代码雨特效

    JS实现代码雨特效

    这篇文章主要为大家详细介绍了JS实现代码雨特效,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-01-01
  • JS中正则表达式只有3种匹配模式(没有单行模式)详解

    JS中正则表达式只有3种匹配模式(没有单行模式)详解

    下面小编就为大家带来一篇JS中正则表达式只有3种匹配模式(没有单行模式)详解。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2016-07-07
  • 轻松实现Bootstrap图片轮播

    轻松实现Bootstrap图片轮播

    这篇文章主要介绍了全面解析Bootstrap图片轮播效果,Bootstrap提供了carousel插件,实现图片轮播非常方便,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2016-01-01
  • 前端深入理解Typescript泛型概念

    前端深入理解Typescript泛型概念

    这篇文章主要介绍了前端深入理解Typescript泛型概念,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-03-03
  • 微信小程序-消息提示框实例

    微信小程序-消息提示框实例

    本篇文章主要介绍了微信小程序-提示框,现在分享给大家,也给大家做个参考。感兴趣的小伙伴们可以参考一下。
    2016-11-11

最新评论