`
fengqx
  • 浏览: 98108 次
  • 性别: Icon_minigender_1
  • 来自: 青岛
社区版块
存档分类
最新评论

js闭包问题

阅读更多
什么是闭包(closures)
闭包就是内部函数通过某种方式使其可见范围超出了其定义的范围,这就产生了一个在其定义范围内的闭包.
一 、内部函数(inner function)
javascript是支持内部函数申明(inner function declaration)的编程语言,

内部函数就是在另一个函数的内部定义,如
function outerFun(){
  function innerFun(){
    alert('hello');
  }
}

函数innerFun就是内部函数, 它在函数outerFun范围内是可见的,
也就是说innerFun函数的命名空间(namespace)是在outerFun范围之内.

正确调用
function outerFun(){
  function innerFun(){
    alert('hello');
  }
  innerFun();
}
outerFun(); alerts hello
错误调用(error)
function outerFun(){
  function innerFun(){
    alert('hello');
  }
}
innerFun();

那么如果我想在函数outerFun外面调用函数innerFun,我该如何做呢
做法1
var globVar;
function outerFun() {
  function innerFun() {
   alert('hello');
  }
  globVar = innerFun;
}
outerFun();
globVar();
做法2
function outerFun() {
  function innerFun() {
    alert('hello');
  }
  return innerFun ;
}
var globVar = outerFun();
globVar();

做法3
function outerFun() {
  function innerFun() {
    alert('hello');
  }
  return {'innerFun'innerFun} ;
}
var globVar = outerFun();
globVar.innerFun();

以上三种做法把内部函数的可见范围扩大了, 其中最后一种做法是把内部函数当做匿名对象{'innerFun'innerFun}的属性,并随之一起返回.

从中看到在javascript里面,函数名称可以当作是一种引用变量,类似于c里面指针的概念,在这里,随着程序的执行
会产生两个引用变量指向内部函数innerFun,一个是globVar(第三种做法是globVar.innerFun),另一个是其函数自身innerFun,
只不过这两个变量的可见范围不一样,即命名空间不一样.
javascript垃圾回收器会在函数最后一个引用变量被废弃后,释放其所占用的内存.


二 变量范围
例1 内部函数变量
在内部函数内申明的变量其可见范围就在其函数内
function outerFun() {
   function innerFun() {
      var innerVar = 0;
      innerVar++;
      alert(innerVar);
   }
   return innerFun;
}
var globVar = outerFun();
globVar();  Alerts 1
globVar();  Alerts 1
var innerVar2 = outerFun();
innerVar2();  Alerts 1
innerVar2();  Alerts 1

每一次内部函数调用,一个新的innerVar变量都被创建,所以结果显示都是1


例2 内部函数引用全局变量(global variables)

var globVar = 0;
function outerFun() {
  function innerFun() {
    globVar++;
    alert(globVar);
  }
  return innerFun;
}
var globVar = outerFun();
globVar();  Alerts 1
globVar();  Alerts 2
var globVar2 = outerFun();
globVar2();  Alerts 3
globVar2();  Alerts 4
每一次内部函数的调用,全局变量都增加1,所以显示结果都是依次递增.

例3 内部函数引用外部函数变量
 
  function outerFun() {
     var outerVar = 0;
     function innerFun() {
       outerVar++;
       alert(outerVar);
     }
     return innerFun;
  }
 
  var globVar = outerFun();
  globVar();  Alerts 1
  globVar();  Alerts 2
  var globVar2 = outerFun();
  globVar2();  Alerts 1
  globVar2();  Alerts 2
  注意在第2次调用outerFun()的时候重新创建了一个新的变量outerVar,所以显示结果是1,2,1,2

什么是闭包(closures)呢, 就是内部函数通过某种方式使其可见范围超出了其定义的范围, 如上例3,globVar 就是函数outerFun()的一个闭包,
而闭包产生的时机就是这句代码var globVar = outerFun(); 再看上例3,闭包globVar第一次调用之后,变量outerVar 值 1 还是在内存
中,没有回收,因为闭包globVar第二次调用的时候其值已经递增为2 ,只要globVar没有被废弃掉,则outerVar的值就会一直存在. 像outerVar
这样的变量称其为自由变量

三 闭包与面向对象编程之间的联系
   看看下面的例子
   function outerFun() {
     var outerVar = 0;
     function innerFun() {
     outerVar++;
     alert(outerVar);
     }
     function innerFun2() {
       outerVar = outerVar + 2;
       alert(globVar);
     }
     return {'innerFun' innerFun, 'outerFun2' outerFun2};
   }

   var globVar = outerFun();
   globVar.innerFun();  Alerts 1
   globVar.innerFun2();  Alerts 3
   globVar.innerFun();  Alerts 4
   var globVar2 = outerFun();
   globVar2.innerFun();  Alerts 1
   globVar2.innerFun2();  Alerts 3
   globVar2.innerFun();  Alerts 4
  
   outerFun()可以看成是类,globVar,globVar2可以看成是两个实例,实例变量就是outerVar,并且是private. 函数innerFun(),
   innerFun2()就是实例方法. 
  
   可能这还不是很清楚,在看下面的例子
   function Boy(){
      var name;
      function setName(vName){
        name = vName;
      }
      function getName(){
        return name;
      }
      return {'setName' setName, 'getName' getName};
   }
   var boy1 = Boy();
   boy1.setName(zhuzhenhua);
   alert(boy1.getName());
  
分享到:
评论

相关推荐

    【JavaScript源代码】详解JavaScript闭包问题.docx

    详解JavaScript闭包问题  闭包是纯函数式编程语言的传统特性之一。通过将闭包视为核心语言构件的组成部分,JavaScript语言展示了其与函数式编程语言的紧密联系。由于能够简化复杂的操作,闭包在主流JavaScript库...

    JS闭包可被利用的常见场景

    JS闭包可被利用的常见场景。值得保留的文档。值得一看

    Javascript 闭包完整解释

    Javascript 闭包完整解释

    js闭包详细讲解

    js闭包的详细讲解

    js闭包个人理解

    js闭包的个人理解

    js闭包写法学习demo

    js闭包写法学习demo

    JavaScript闭包

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

    尚硅谷——JavaScript闭包

    JavaScript闭包 JavaScript闭包 JavaScript闭包 JavaScript闭包

    js闭包理解之倒计时

    网上大多只是讲解js闭包的大概意思和小例子,但真实用于项目中又该怎么用呢。这个例子真实的告诉大家在项目中的用法

    JS闭包经典

    闭包一点即通 闭包(closure)是Javascript语言的一个难点,也是它的特色,很多高级应用都要依靠闭包实现。

    js闭包的理解以及作用.docx

    JS闭包理解 什么是闭包? 当有一个函数想要访问另一个函数内部的变量,这个是访问不了的。所有我们要用闭包来访问。所以简单的来说,闭包就是连接函数内部和外部的一座桥梁,通过闭包我们能够访问其他函数内部的...

    javascript 闭包

    闭包(closure)是Javascript语言的一个难点,也是它的特色,很多高级应用都要依靠闭包实现。

    JavaScript闭包函数

    闭包是ECMAScript (JavaScript)最强大的特性之一,但用好闭包的前提是必须理解闭包。闭包的创建相对容易,人们甚至会在不经意间创建闭包,但这些无意创建的闭包却存在潜在的危害,尤其是在比较常见的浏览器环境下...

    js闭包学习

    javaScript无可争议的是如今前台开发最常用的技术,该文档讲解了javaScript中闭包技术,在jquery中大量的使用了该技术,学好可以更好的理解学习jquery。

    深入理解javascript原型和闭包

    深入理解javascript原型和闭包(01)——一切都是对象 深入理解javascript原型和闭包(02)——函数和对象的关系

    javascript闭包详解中文word版

    资源名称:javascript闭包详解 中文word版   内容简介: Javascript中有几个非常重要的语言特性——对象、原型继承、闭包。其中闭包 对于那些使用传统静态语言C/C 的程序员来说是一个新的...

    JS中的闭包理解。。。。。。。。

    js闭包 • 每个函数都有一个包含词法环境的执行上下文,它的词法环境确定了函数内的变量赋值以及对外部环境的引用。看上去函数“记住”了外部环境,但其实上是这个函数有个指向外部环境的引用。这就是“闭包”的...

    【JavaScript源代码】js闭包和垃圾回收机制示例详解.docx

    js闭包和垃圾回收机制示例详解  目录 前言 正文  1.闭包  1.1闭包是什么?  1.2闭包的特性 1.3理解闭包  1.4闭包的主要实现形式  1.5闭包的优缺点  1.6闭包的使用  2.垃圾回收机制 总结  前言  正文  ...

    js代码-JS闭包以及优化

    js代码-JS闭包以及优化

    javascript深入理解js闭包.docx

    javascript深入理解js闭包.docx

Global site tag (gtag.js) - Google Analytics