闭包

1. 概念

闭包是指那些能够访问独立(自由)变量的函数 (变量在本地使用,但定义在一个封闭的作用域中)。换句话说,这些函数可以“记忆”它被创建时候的环境。

2.词法作用域

首先看下面的代码:

function init() {
  var name = "Mozilla";
  function displayName() {
    alert(name);
  }
  displayName();
}
init();

这里的函数执行结果是'Mozilla',原因是当函数执行到alert(name)的时候,首先在函数内部查找变量name,如果没有则会按照作用域链一层一层向上查找次序是这样的

function displayName ->function init -> global

所以,init函数执行的时候会弹出'Mozilla',因为init函数中定义了变量name

2.闭包

接下来来看这个例子

function makeFunc() {
  var name = "Mozilla";
  function displayName() {
    alert(name);
  }
  return displayName;
}

var myFunc = makeFunc();
myFunc();

这段代码特殊的地方在于,函数调用后变量name没有被垃圾回收机制回收。 原因是myFunc 变成一个 闭包 了。
闭包是一种特殊的对象。它由两部分构成:函数,以及创建该函数的环境。环境由闭包创建时在作用域中的任何局部变量组成。在我们的例子中,myFunc 是一个闭包,由 displayName 函数和闭包创建时存在的 "Mozilla" 字符串形成。

为什么myFunc中的name在函数执行后没有被销毁? 这个就要从JS的垃圾回收机制来说起了

3.垃圾回收机制

目前JS的垃圾回收机制无非就是两种:
1.标记清除(make-and-sweep)
2.引用计数(reference counting)

1.标记清除:标记清除简单的来说就是给各个变量名打上 YES or NO的标签以供JS引擎进行处理(当然打什么标签自己理解即可)。在和执行上下文类似的的环境中当变量名称进入环境的时候,那么变量会被打上 YES。一般来说是绝对不会释放被打上 YES 标签的变量内存的,一旦变量在出了该环境时,变会被打上 NO 标签(和作用域貌似有点像),JS引擎会在一定时间间隔或者设置的时间来进行扫描,对NO标签的进行剔除以释放其内存。

2.引用计数
一般来说,引用计数的含义是跟踪记录每个值被引用的次数。当声明一个变量并将一个引用类型值赋给该变量时,则这个值的引用次数便是1,如果同一个值又被赋给另一个变量,则该值的引用次数加1,相反,如果包含对这个值引用的变量又取得了另一个值,则这个值的引用次数减1。当这个值的引用次数为0时,说明没有办法访问到它了,因而可以将其占用的内存空间回收

在Javascript中,如果一个对象不再被引用,那么这个对象就会被回收(打上了NO标记)。如果两个对象互相引用,而不再被第3者所引用,那么这两个互相引用的对象也会被回收。因为函数a被b引用,b又被a外的c引用,这就是为什么函数a执行后不会被回收的原因。

4 继续闭包

依然是这个例子

function makeFunc() {
  var name = "Mozilla";
  function displayName() {
    alert(name);
  }
  return displayName;
}

var myFunc = makeFunc();
myFunc();

实际上上面的函数表达式可以看做

makeFunc()();

在调用函数makeFunc的时候,由于他的返回值是函数displayName,那么函数displayName并不会在执行完毕之后直接销毁,而是被JS认为一直处于被调用状态。

来看下面一个例子:

function makeAdder(x) {
  return function(y) {
    return x + y;
  };
}

var add5 = makeAdder(5);
var add10 = makeAdder(10);

console.log(add5(2));  // 7
console.log(add10(2)); // 12

实际上,最后的两个console.log()依然可以看做

makeAdder(5)(2);
makeAdder(10)(2);

在调用makeAdder的同时它的return返回的函数也在被调用。

5.闭包的神奇用法

5.1 一些设置

function setSize(num){
    //在这里把计算函数return出去
    return function(){
        doucment.body.style.fontSize = size + "px";
    }
}

//调用一下试试?
var fontSet12 = makeAdder(12);
makeAdder(14);
makeAdder(16);

//于是你可以这样绑定函数了
var Btn12 = document.getElementById('btn-12');
btn12.addEventListener('click',fontSet12);

5.2 私有方法

你可以用这个方法模拟面向对象中的私有方法,怕不怕!

var count = function(){
    //首先定义一个自执行函数
     var privateCounter = 0;
  function changeBy(val) {
    privateCounter += val;
  }
  return {
    increment: function() {
      changeBy(1);
    },
    decrement: function() {
      changeBy(-1);
    },
    value: function() {
      return privateCounter;
    }
  }   
};

//或许我们见过另外一种形式的写法

var count = function(){
    var privateCounter = 0;

    function changeBy(val) {
        privateCounter += val;
    }

    function increment (){
        changeBy(1);
    }

    function decrement () {
        changeBy(-1);
    }

    function value() {
        return privateCounter;
    }

    return {
        increment:increment,
        decrement:decrement,
        value: value
    }
};

然后我们做一些操作

var con1 = count();
var con2 = count();

查看count的value返回值

console.log(con1.value()),con2.value();     //0,0

分别执行不同的操作

con1.decrement();
con2.increment();

再次查看

console.log(con1.value(),con2.value());     //-1,1

5.3 函数调用后的指向问题:

问题:那么函数和对象都是引用类型,函数调用之后返回值是对象的时候,多次调用这个函数的时候他们的返回值不是指向同一个地址的么?
解答:
可以把函数看做是一个类,那么函数调用就是这个类的实例化,实例化之后的指向自然不一样

function a(){}
var a1 = new a;
var a2 = new a;

console.log(a1 == a2);      //false
image_1atvnvaiqkeknlb1j3bcg45339.png-6.5kB
image_1atvnvaiqkeknlb1j3bcg45339.png-6.5kB

这里返回值也是false

6.闭包的性能

如果不是因为某些特殊任务而需要闭包,在没有必要的情况下,在其它函数中创建函数是不明智的,因为闭包对脚本性能具有负面影响,包括处理速度和内存消耗。
闭包会造成内存泄露!

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,874评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,102评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,676评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,911评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,937评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,935评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,860评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,660评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,113评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,363评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,506评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,238评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,861评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,486评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,674评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,513评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,426评论 2 352

推荐阅读更多精彩内容