JS中的闭包

什么是闭包?

(function(){
  var local = '你好'
  function fn(){
    console.log(local)
  }
})()

上面的代码就形成了一个闭包:
「函数」和「函数内部能访问到的变量」的总和,就是一个闭包
所以,可以这么说,JS中所有的函数都是闭包。因为JS中所有的函数都可以访问全局变量,那么全局变量和函数就形成了闭包
闭包不是故意弄出来的东西,而是JS函数作用域的副产品。

函数作用域

ES 5 中只有两种作用域,全局作用域和函数作用域。函数内部的变量可以读取全局作用域中的变量,而函数外部不能读取函数内部定义的变量。而闭包,就是沟通全局作用域和函数作用域之间的桥梁,全局作用域和函数作用域通过闭包来实现变量的连接。

闭包的一般形式

常见的使用闭包的一般形式是使用自执行函数 或 函数里面套函数,目的是为了提供一个函数作用域,隔开函数作用域和全局作用域,达到隐藏变量的目的。使用闭包将变量声明放到函数作用域中,除了这个函数的其他地方就不能访问到这个变量,变量就隐藏了。

闭包的用途

  1. 读取函数内部变量
function foo(){
  var xx = '你好'
  return function(){
    return xx
  }
}
// 上面这段代码中,函数 foo 返回的匿名函数与 foo 中的变量 xx 形成了一个闭包。
// foo 返回一个匿名函数,该匿名函数读取 foo 作用域中的变量,之后返回该变量。
// 调用函数 foo ,返回一个匿名函数,再调用该匿名函数,就可以读取 foo 中的变量。

var getVariable = foo()
getVariable()     //  返回  '你好'
console.log(xx)   // 报错,xx 没有被定义
  1. 让变量保持在内存中,不被垃圾回收机制回收
function foo(xx){
  return function(){
    return xx++
  }
}
// 上面代码中,使用闭包,多创建了一个匿名函数的作用域来保存变量 xx 的值。
// 在函数 foo 执行之后,xx 的值在匿名函数中被保留下来,不会因为没有被引用而被内存回收 

var fn = foo(1)
fn()   // 1,每一次调用 fn,其实都是在调用 fn(foo 中返回的匿名函数) 作用域中的xx
fn()   // 2,每一次调用 fn,其实都是在调用 fn(foo 中返回的匿名函数) 作用域中的xx
fn()   // 3,每一次调用 fn,其实都是在调用 fn(foo 中返回的匿名函数) 作用域中的xx
var li = document.querySelectorAll('li')
for(var i = 0; i < li.length; i++){
  li[i].onclick = function(){
    console.log(i)
  }
}
// 上面这段代码,每一次点击 li, 都会打印出 5
// 这是因为上面的 i,用的都是同一个作用域下的变量 i,这个作用域之中的 i,在经过 for 循环之后,值会变成 5。

// 而想要让代码达成我们想要的效果,就需要为每一个变量 i 创建一个作用域,
// 使打印出来的每一个变量 i 的作用域不同,就是打印出不同的 i,使用闭包就可以做到。
var li = document.querySelectorAll('li')

for(var i = 0; i < li.length; i++){
  li[i].onclick = (function(i){
    return function(){
      console.log(i)
    }
  })(i)
}
// 上面代码新增了一个函数作用域,用来保存变量 i 的值。新增的函数返回要执行的回调函数,
// 又因为 ‘click’事件是直接执行的,所以,要将返回的函数直接执行,使用自执行函数的方式,来执行返回的回调函数。
// 这时调用的回调函数,每一个 i 都是独立函数作用域之中的 i,它们的值都不一样。
  1. 封装私有变量、方法
function fn(a){
  var b = 'hi'
  
  function add(){
    return a++
  }
  
  function reduce(){
    return a--
  }
  
  function getA(){
     return a
  }
  
  function getB(){
    return b
  }
  
  function modefyB(x){
    b = x
    return b
  }
  
  return {
    getA: getA,
    add: add,
    reduce: reduce,
    getB: getB,
    modefyB: modefyB,
  }
}

// 上面的代码封装了两个私有变量和4个私有方法。
// 一个私有变量是参数传入的,一个私有变量是自身定义的。
// 通过私有方法来对私有变量进行操作。

var f1 = fn(1)
var f2 = fn(10)
// f1 和 f2 是两个不同的对象,拥有各自的私有变量和私有方法,分别对 f1 和 f2 进行操作,并不会影响另一个。

f1.getA()  // 1
f1.getB() // 'hi'
f1.modefyB('hello')
f1.getB()  // 'hello'

f2.getA()  // 10
f2.getB()  // 'hi'
f2.modefyB('放心,我不会改变对象f1的。不信你试试?')
f2.getB()  // '放心,我不会改变对象f1的。不信你试试?'

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

推荐阅读更多精彩内容

  • 首先,js闭包对于哪怕是有很多年前端开发经验的人,也是很晦涩难懂的东西. 所以,我不敢保证能把你说明白,但是如果你...
    火锅伯南克阅读 403评论 0 1
  • 在上一篇文章“执行环境和作用域”中,我试着梳理了执行环境和作用域的关系。但实际上,文章中并没有提到作用域,而是介绍...
    海痕阅读 237评论 0 0
  • 今天研究了一波js中的闭包,分享一下自己的理解。 一、变量的作用域 要理解闭包,首先必须理解Javascript特...
    阿布_0caf阅读 243评论 0 2
  • 前两天和老同学吃饭,大家回顾这几年自己的发展,有一个共识:大学刚毕业前几年,因为不懂职场规则,走了很多弯路,如果当...
    王二宝阅读 145评论 0 0