JS 基础知识点及常考面试题(二)2 闭包

闭包 (closure)

概念

  • 闭包是函数(内层)和声明该函数的词法环境的组合
  • 闭包 外层返回内层 总之 内层在外层结束访问外层局部变量的现象
  • 注:异步是异步 闭包是闭包 无关系
    例:下异步非闭包,存在bug,在本文末与解释
for (var i = 1; i <= 5; i++) {
  setTimeout(function timer() {
    console.log(i)
  }, i * 1000)
}
  • 闭包三要素 : 函数嵌套函数 函数返回函数 内层函数能访问外层局部变量
    对于函数返回函数说法不准确 ,闭包其实是词法环境
  • 即外部函数存在的意义是提供内部函数需求的所有局部变量和定义内部函数
    该词法环境称之为闭包
  • JavaScript中的函数会形成闭包。 闭包是由函数以及创建该函数的词法环境组合而成。这个环境包含了这个闭包创建时所能访问的所有局部变量。
  • 注:闭包只是一个词法环境,如上解释。
  • 对闭包的合理利用 指当内部函数在执行前被外部函数返回,即外部函数执行返回值是该词法环境的闭包(包括了所有局部变量和内层函数),在外部函数执行完毕仍然能访问该词法环境的局部变量的现象
  • 在闭包中会一直引用变量(是引用而不是副本),直到其断开连接不再引用,在内存中的闭包就可以得到释放。
闭包理解和合理利用
  • 内部函数在执行前被外部函数返回

  • 内部函数和外部局部变量组成闭包,在外部函数执行完毕销毁,内部函数和变量组合词法环境被外部调用时还能够访问到外部函数的变量,

  • 普通函数嵌套,当子函数内部使用自己的局部变量,则随作用域和函数执行过程,变量被正常销毁,js的垃圾回收机制。
    但是当外部函数返回内层函数时,由闭包的存在,外部函数执行完毕,内部函数仍能访问该词法环境的外部函数的变量,该词法环境的变量仍然未被销毁。

  • 可以多个内部函数共享一个词法环境

  • 闭包词法环境的独立性 共享词法环境的闭包,改变调用的变量,会改变闭包的词法环境,故而一个闭包中对变量的修改,不会影响另一个闭包中的变量。即共享变量的词法环境的闭包,创建多个实例,数据的维护具有独立性,互不影响。

理解

  • 理解 1、闭包是内层函数和创建内层时所能访问的所有外层函数的局部变量所组成的词法环境
  • 等价
  • 未定论点:MDN文档 闭包:JavaScript中的函数会形成闭包。 闭包是由函数以及创建该函数的词法环境组合而成。这个环境包含了这个闭包创建时所能访问的所有局部变量。
    理解 ?每个函数和外层局部变量的词法环境组成闭包,当外层函数返回内层函数时,可以利用闭包的特性(词法环境,外部访问)实现不同的效果,如循环条件的var与定时器的冲突,闭包于实例中的数据维护的独立性等
  • 理解 2、对闭包的利用是 闭包(内层相关的词法环境)在外层函数执行完毕被销毁时,内层的实例词法环境仍然能访问到该词法环境的局部变量的现象。即该实例下的局部变量待机
  • 理解 3、 封装的外层函数内的闭包,共享相同局部变量的词法环境,可通过创建不同的实例,利用统一闭包,传递不同变量,形成不同词法环境,即不同实例下闭包算法互不影响,数据维护具有独立性。
  • 理解4、在闭包中会一直引用变量(是引用而不是副本),直到其断开连接不再引用,在内存中的闭包就可以得到释放。
    案例: MDN 闭包计数器
var makeCounter = function() {
  var privateCounter = 0;
  function changeBy(val) {
    privateCounter += val;
  }
  return {
    increment: function() {
      changeBy(1);
    },
    decrement: function() {
      changeBy(-1);
    },
    value: function() {
      return privateCounter;
    }
  }  
};

var Counter1 = makeCounter();
var Counter2 = makeCounter();
console.log(Counter1.value()); /* logs 0 */
Counter1.increment();
Counter1.increment();
console.log(Counter1.value()); /* logs 2 */
Counter1.decrement();
console.log(Counter1.value()); /* logs 1 */
console.log(Counter2.value()); /* logs 0 */
  • 闭包的私有方法 模块模式 数据维护的独立性
  • 以这种方式使用闭包,提供了许多与面向对象编程相关的好处 —— 特别是数据隐藏和封装。

About 词法作用域

构造函数

解决情境

一: 循环中使用闭包解决 var 定义函数的问题

问题如下

for (var i = 1; i <= 5; i++) {
  setTimeout(function timer() {
    console.log(i)
  }, i * 1000)
}

需求: 每秒console当前循环的i
效果:间隔1秒打印6
原因:
1、定时器内部异步
2、对于循环 i 1 => 6 循环5次 最终i结果为6
3、异步 即 循环完毕 6个定时器的异步 其中定时器时间未进入异步,挂载的是每次循环的i
4、多个异步在循环完毕后同时进行,即基于的基准时间点是同一个,因此延迟时间1s,2s,3s,4s,5s显示的效果为间隔1s
解决:

for (var i = 1; i <= 5; i++) {
  ;(function(j) {
    setTimeout(function timer() {
      console.log(j)
    }, j * 1000)
  })(i)
}

思路:

  • 自执行函数传参当前i
  • 闭包的定时器异步获取参数
  • 因为异步,闭包的词法环境保留 定时器取到真实的j
  • 同时异步 仍然是基于同一个时间点,即间隔1s打印1-5


使用 setTimeout 的第三个参数,这个参数会被当成 timer 函数的参数传入。

for (var i = 1; i <= 5; i++) {
  setTimeout(
    function timer(j) {
      console.log(j)
    },
    i * 1000,
    i
  )
}


ES6使用 let 定义 i 了来解决问题了,简洁

for (let i = 1; i <= 5; i++) {
  setTimeout(function timer() {
    console.log(i)
  }, i * 1000)
}


对于数组对象的iforEach,可以顺序打印数组的值,但是无间隔
LINK

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