JS 中的闭包是什么?

本文为饥人谷讲师方方原创文章,首发于 前端学习指南html

大名鼎鼎的闭包!面试必问。
请用本身的话简述前端

  1. 什么是「闭包」。
  2. 「闭包」的做用是什么。

首先来简述什么是闭包

假设上面三行代码在一个当即执行函数中(为简明起见,我就不写当即执行函数了,影响读者理解)。面试

评论里没看完就说我写得有问题的,请看清楚哦:编程

上面三行代码在一个当即执行函数中。ruby

三行代码中,有一个局部变量 local,有一个函数 foo,foo 里面能够访问到 local 变量。微信

好了这就是一个闭包:闭包

「函数」和「函数内部能访问到的变量」(也叫环境)的总和,就是一个闭包。编程语言

就这么简单。函数

有的同窗就疑惑了,闭包这么简单么?学习

「我据说闭包是须要函数套函数,而后 return 一个函数的呀!」

好比这样:

function foo(){
  var local = 1
  function bar(){
    local++
    return local
  }
  return bar
}

var func = foo()
func()

这里面确实有闭包,local 变量和 bar 函数就组成了一个闭包(Closure)。

为何要函数套函数呢?

是由于须要局部变量,因此才把 local 放在一个函数里,若是不把 local 放在一个函数里,local 就是一个全局变量了,达不到使用闭包的目的——隐藏变量(等会会讲)。

这也是为何我上面要说「运行在一个当即执行函数中」。

有些人看到「闭包」这个名字,就必定以为要用什么包起来才行。其实这是翻译问题,闭包的原文是 Closure,跟「包」没有任何关系。

因此函数套函数只是为了造出一个局部变量,跟闭包无关。

为何要 return bar 呢?

由于若是不 return,你就没法使用这个闭包。把 return bar 改为 window.bar = bar 也是同样的,只要让外面能够访问到这个 bar 函数就好了。

因此 return bar 只是为了 bar 能被使用,也跟闭包无关。

闭包的做用

闭包经常用来「间接访问一个变量」。换句话说,「隐藏一个变量」。

假设咱们在作一个游戏,在写其中关于「还剩几条命」的代码。
若是不用闭包,你能够直接用一个全局变量:

window.lives = 30 // 还有三十条命

这样看起来很不妥。万一不当心把这个值改为 -1 了怎么办。因此咱们不能让别人「直接访问」这个变量。怎么办呢?

用局部变量。

可是用局部变量别人又访问不到,怎么办呢?

暴露一个访问器(函数),让别人能够「间接访问」。

代码以下:

!function(){

  var lives = 50

  window.奖励一条命 = function(){
    lives += 1
  }

  window.死一条命 = function(){
    lives -= 1
  }

}()

简明起见,我用了中文 :)

那么在其余的 JS 文件,就可使用 window.奖励一条命() 来涨命,使用 window.死一条命() 来让角色掉一条命。

看到闭包在哪了吗?

闭包究竟是什么?

五年前,我也被这个问题困扰,因而去搜了 stackoverflow 并总结下来。你在百度搜闭包,那篇《JavaScript闭包——懂不懂由你,反正我是懂了》就是我写的。当时我仍是新手,一直不理解为何你们口中的闭包这么模糊、这么琢磨不定呢。
咱们从新来审视一下闭包的代码:

第一句是变量声明,第二句是函数声明,第三句是 console.log。
每一句我都学过,为何合起来我就看不出来是闭包?

我告诉你答案,你根本不须要知道闭包这个概念,同样可使用闭包!

闭包是 JS 函数做用域的副产品。

换句话说,正是因为 JS 的函数内部可使用函数外部的变量,因此这段代码正好符合了闭包的定义。而不是 JS 故意要使用闭包。

不少编程语言也支持闭包,另外有一些语言则不支持闭包。

只要你懂了 JS 的做用域,你天然而然就懂了闭包,即便你不知道那就是闭包!

所谓闭包的做用

若是咱们在写代码时,根本就不知道闭包,只是按照本身的意图写,最后,发现知足了闭包的定义。

那么请问,这算是闭包的做用吗?

这个问题,留给你思考。

关于闭包的谣言

闭包会形成内存泄露?

错。

说这话的人根本不知道什么是内存泄露。内存泄露是指你用不到(访问不到)的变量,依然占居着内存空间,不能被再次利用起来。

闭包里面的变量明明就是咱们须要的变量(lives),凭什么说是内存泄露?

这个谣言是如何来的?

由于 IE。IE 有 bug,IE 在咱们使用完闭包以后,依然回收不了闭包里面引用的变量。

这是 IE 的问题,不是闭包的问题。参见司徒正美的这篇文章

一个小经验

编程界崇尚以简洁优雅惟美,不少时候

若是你以为一个概念很复杂,那么极可能是你理解错了。

完。

加微信号: astak10或者长按识别下方二维码进入前端技术交流群 ,暗号:写代码啦

每日一题,每周资源推荐,精彩博客推荐,工做、笔试、面试经验交流解答,免费直播课,群友轻分享... ,数不尽的福利免费送