`
549265480
  • 浏览: 29699 次
  • 性别: Icon_minigender_1
  • 来自: 成都
文章分类
社区版块
存档分类

【初窥javascript奥秘之闭包】叶大侠病都好了,求不踩了:)

 
阅读更多

前言

咳咳,叶大侠病了,昨天晚上回家时候在车上就不舒服,果然回来就挂了,本来还想今天接着上班撑下去的。但是昨天又看到一个IT巨子挂了,所以果断请了一个假!!!

但是早上7.00左右就迷迷糊糊的醒了,于是我在想我是不是该“身残志坚”一番。。。。

咳咳,以上玩笑,我们最近一起学习了很多CSS的东西,相信大家的CSS水平必定提高了吧???所以我们接下来一段时间来看看javascript吧,今天我们一起来看看闭包这个家伙!

本文参考:

http://www.cnblogs.com/TomXu/

http://www.cnblogs.com/lhb25/archive/2011/09/06/javascript-scope-chain.html

http://www.cnblogs.com/frankfang/archive/2011/08/03/2125663.html

http://www.cnblogs.com/xiaotie/archive/2011/08/03/2126145.html

执行上下文(execution context)

又是上下文。。。无论是曾经学习.net时候的httpcontext,还是最近学习的block formatting context,都是上下文!

我们可以感觉到上下文的存在,就如感觉得到空气一样,我们却将它拿不出来,于是我们来看看执行上下文吧。

每次当控制器转到ecmascript可执行代码的时候,就会进入一个执行上下文。
活动的执行上下文组在逻辑上为一个堆栈,堆栈最底部一定是全局上下文(global context),顶部便是当前上下文

这个句话是很有玄机的哦,比如我们在一个函数环境中,我们首先用的是当前变量,若是没有就在父级函数中找,最后就在window中找。

PS:我们在写博客时候,中文没有问题,突然打一个字母上,你会发现整行都下移了1/2px,各位知道神马原因么???:)

eval在执行时候会产生一个调用上下文的东东(calling context),所以eval中定义的变量会影响函数环境中的变量。

作用域

闭包有作用域链有很多说不清道不明的关系,所以我们还是先看看作用域吧。

任何语言都会有作用域的概念,作用域便是变量与函数可访问范围,简称变量与函数的生命周期。
javascript中分为全局与局部两种,直白点window与function

var name = '叶小钗';
function func() {
    var id = '刀狂剑痴';
    function alertId() {
        alert(id);
    }
    alertId();
}
alert(name);//叶小钗
alert(id);//错误
func(); //刀狂剑痴
alertId();//错误

此处name属于window属性便是全局的(有些浏览器自带name属性小心陷阱!)

func也是全局的,但是func这个作用域链包含了全局的,所以他可以访问外部的,但是外部不能访问进来。

链子。。。

在js中什么都是对象,函数和其它对象一样拥有很多属性,其中一个比较特殊的属性是:

[[scope]],scope翻译过来就是作用域,该属性包含了函数被创建的作用域中的对象的集合:

于是传说中的作用域链诞生了,他决定了哪些数据能被哪些函数访问。

当一个函数创建后,他的作用域链会被创建此函数的作用域中可访问的数据对象填充:

function add (a, b) {
    var sum = a + b;
    return sum;
}

在函数add创建时,他的作用域链中会填入一个全局对象,我们可以想象为window被压入了其[[scope]]

函数add的作用域在调用时会被用到,此时会创建一个“运行时上下文(execution context)”的内部对象,
他定义了函数执行时的环境,每个运行时上下文都有自己的作用域链,用于标识符解析,
当execution context创建时,而他的作用域链初始化为当前运行函数的[[scope]]所包含的对象。

所有相关的东东会按照其在函数中出现的顺序被复制到运行时上下文的作用域链中,他们共同组成了“活动对象(activation object)”,该对象就包含了函数的所有局部变量,参数,以及this,然后会排在我们作用域堆栈的顶端,我们可以最先获取,在运行时上下文结束(函数执行完毕)后,GC便会回收其空间。

在函数执行过程中,遇到一个变量便会经历一次标识符解析以决定从哪里获取以及存储数据。

这个过程首先从我们的“堆栈”顶端开始搜索,也就是活动对象的开始,若是有该变量便获取值,

若是没有则向下,知道window为止,没有就报错!

内存溢出

是想我们的浏览器内存若是搞满了,自然会浏览器崩溃,这是我们闭包之中需要考虑到的东西,因为他会导致我们一些东西无法GC。

闭包,神奇的魔法

闭包是神奇的魔法,因为他干了很多“坏事”,达到了出人意料的结果。

所谓闭包,便是function嵌套function,内部function可以访问外部的变量。
若是外部函数返回了内部函数,那么就闭了一个包
 1 function a() {
 2     var name = '叶小钗';
 3     function alertName() {
 4         alert(name);
 5     }
 6     return alertName;
 7 }
 8 
 9 var func = a();
10 func();//叶小钗

这个家伙是个坏孩子,怎么说呢,坏孩子招人疼啊!

我们看第9行,按我们之前的理解a执行结束后,整个活动对象是不是该被销毁?是不是与a有关的东西都该GC呢?

确实是这样的,但是我们这里就产生了一个闭包,阻止了其销毁,因为他里面的变量被用到了。

这里的一个事实便是外部的func访问到了a内部的变量!

一个经典的例子:

 1 function outer() {
 2     var o = {};
 3     for (var i = 0; i < 10; i++) {
 4         o[i] = function () {
 5             alert(i);
 6         }
 7     }
 8     return o;
 9 }
10 var funcs = outer();
11 for (var k in funcs) {
12     funcs[k]();
13 }
14 var s = '';

我们来看这个,大家应该都比较熟悉了,我们知道他全部会打印10。

这是因为我们将outer给予funcs时候,outer的任务就完成啦,他现在内部的i便是10,我们现在再来调用函数,所有的i自然是指向一个[[scope]],所以变量i就是10!!!

要怎么解决大家也一目了然:

 1 function outer() {
 2     var o = {};
 3     for (var i = 0; i < 10; i++) {
 4         o[i] = (function (i) {
 5             alert(i);
 6         })(i)
 7     }
 8     return o;
 9 }
10 var funcs = outer();
11 for (var k in funcs) {
12     funcs[k]();
13 }
14 var s = '';

相当于每次i是以一个副本的方式重新复制了一番,这个是可以达到我们的目的,但是有时候会对性能有一点影响,我们还是需要注意。

我们这里就不扯远了,在扯远的话我估计也露馅了。。。呵呵,最后我们来说一点应用吧。

闭包的应用

jquery时如何神奇的使用闭包的我现在还看不透,那是我1年后的任务,我这不关注他。

我这里说的应用时原来项目过程中遇到的,所以没有代码,我这边说下问题吧:

我们点击一个按钮,会向服务器发一个ajax请求,其中包含一些data,然后数据返回回来后我们又会用到开始传过去的一些data中的敏感信息,比如id。

这个时候一种比较笨的方法就是服务器给我们传回来,但是闭包出现后,我们便可以直接在函数中定义函数,而使用我们之前的data即可。

该例子有段时间了,大家看不清楚也没有关系啦。。。。

结语

叶大侠病了,求安慰,求按顶。。。。。

分享到:
评论

相关推荐

    深入理解javascript原型和闭包

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

    理解_JavaScript_闭包

    本文结合 ECMA 262 规范详解了闭包的内部工作机制,让 JavaScript 编程人员对闭包的理解从“嵌套的函数”深入到“标识符解析、执行环境和作用域链”等等 JavaScript 对象背后的运行机制当中,真正领会到闭包的实质。

    javascript 闭包

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

    Javascript 闭包完整解释

    ### JavaScript闭包完整解释 #### 一、闭包的基本概念 **闭包**是一个非常重要的JavaScript概念,它指的是一个函数能够记住并访问其外部作用域中的变量的能力,即使该函数在其外部作用域之外被调用也是如此。具体...

    JavaScript闭包

    Javascript中有几个非常重要的语言特性——对象、原型继承、闭包。其中闭包 对于那些使用传统静态语言C/C++的程序员来说是一个新的语言特性。本文将以例子入手来介绍Javascript闭包的语言特性,并结合一点 ...

    JavaScript闭包函数

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

    深入理解javascript原型和闭包.pdf

    JavaScript原型和闭包是这门语言中两个比较难以理解且与其他面向对象语言区别较大的概念。...无论是原型链和作用域链的理解,还是函数作为一等对象的灵活使用,都是深入学习和应用JavaScript不可或缺的部分。

    JavaScript核心(对象、原型、继承、上下文、闭包、this).pdf

    理解上下文对于理解JavaScript的作用域、this关键字和闭包等概念非常关键。 总结以上知识点,可以看出JavaScript通过原型链实现了继承机制,原型链是对象之间共享属性和方法的一种机制,与传统的类继承方式不同,...

    javascript里的闭包是什么 什么是闭包.zip

    JavaScript中的闭包是一种重要的编程概念,它涉及到函数、作用域和变量持久化等多个核心知识点。闭包的本质是函数能够访问并操作其外部作用域内的变量,即使在其外部作用域已经结束之后仍然能保持对这些变量的访问。...

    数据库求属性集闭包&函数依赖闭包

    声明:以下仅个人观点,若有错误,敬请指正O(∩_∩)O~ 关键点 1) 将函数依赖用multimap,string&gt; 存储,因为函数依赖可能...对每一个子集求其闭包,记为Ri+;然后求Ri+的所有子集记为(Ri+)j;最后做映射Rià(Ri+)j。

    计算NFA中ε闭包

    ### 计算NFA中的ε闭包 #### 一、NFA与ε闭包概念介绍 **非确定有限自动机(NFA)**是一种理论计算模型,它扩展了确定有限自动机(DFA)的概念,允许在某些情况下从一个状态出发到达多个状态。在NFA中,存在一种特殊的...

    JavaScript闭包与立即执行函数详解:特性、实现和应用场景

    内容概要:本文详细讲解了JavaScript中的闭包概念及其特性,并介绍了闭包如何通过内部函数引用外部函数变量来扩展变量作用范围。解释了闭包带来的优点如保护私有变量,但也提到了潜在的风险像可能导致内存泄漏。同时...

    javascript闭包详解中文word版

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

    闭包javascript.pdf

    标签:javascript 闭包 从给定的文件信息来看,主要围绕JavaScript中的闭包概念进行深入探讨。闭包是计算机科学中一个重要的概念,特别是在函数式编程语言中,如JavaScript,它允许一个函数在定义时捕获并记住其...

    深度探讨javascript函数的原型链和闭包

    在JavaScript中,函数扮演着至关重要的角色,它们不仅仅是代码块,更是数据的一种形式。JavaScript的灵活性使得函数可以作为变量、对象的方法甚至构造函数使用。理解函数的原型链和闭包对于深入掌握JavaScript至关...

    javascript闭包的理解

    闭包是一个高级且复杂的话题,它是JavaScript语言的核心特性之一,同时也是一大难点。要想熟练运用JavaScript,掌握闭包的原理和使用方法是非常必要的。 在JavaScript中,变量的作用域分为全局变量和局部变量。全局...

    基于javascript 闭包基础分享

    在JavaScript编程中,闭包是一个核心概念,它允许函数记住并访问所在词法作用域,即使当函数在其词法作用域之外执行时。闭包的特性使得它在JavaScript中既神秘又强大。 首先,我们从闭包的定义谈起。在JavaScript中...

    浅谈JavaScript for循环 闭包_.docx

    JavaScript的闭包机制非常重要,它可以帮助我们更好地理解和使用JavaScript语言。但是,它也可能会引发一些问题,例如for循环中的闭包问题。幸运的是,我们可以使用以上六种方法来解决这个问题。

    跟我学习javascript的闭包

    闭包:是指有权访问另外一个函数作用域中的变量的函数。创建闭包的常见方式就是在一个函数内部创建另外一个函数。 直接上例子 function a(){ var i=0; function b(){ alert&#40;++i&#41;; } return

    尚硅谷——JavaScript闭包

    在尚硅谷学习JavaScript闭包的过程中,我们首先需要了解闭包的作用。闭包是JavaScript中一个独特且重要的概念,它允许函数访问并操作函数外部的变量,这一点体现了JavaScript作用域链的特性。闭包的一个关键用途是...

Global site tag (gtag.js) - Google Analytics