摘要:本质与解析当函数可以记住并访问所在的词法作用域时,就产生了闭包,即使函数是在当前词法作用域之外执行。
本质与解析
当函数可以记住并访问所在的词法作用域时,就产生了闭包,即使函数是在当前词法作用域之外执行。
function outter() { var a = 1 function inner() { console.log(a) } return inner } var accept = outter() accept() // 1
在此例中,将内部函数inner作为返回值,当outter函数执行后,赋值给accept,并且调用accept;实际上调用了inner函数,它在自己定义的词法作用域以外的地方执行。
通常在outter函数执行完成之后,由于js的垃圾回收机制,其内部的作用域会被销毁回收,但是由于闭包的存在,accept持有对inner的引用,而inner声明的位置为outter的内部作用域,因此该作用能够一直存活
面试常见的闭包考题就是循环闭包。例如:
for (var i=0; i<=8; i++) { setTimeout(function () { console.log(i) },1000) }
正常情况下期待每隔一秒输出1~8,但是实际每秒输出一个9,其实不难理解,从执行栈来看,延迟函数会在整个for循环执行完成之后才会执行,即使setTimeout设置的延迟时间是0,同时其被封闭在全局的作用域中的,因此每次得到的都是对全局i变量的引用,实际上只有一个i。
可以通过立即执行函数将作用域封闭起来,同时通过自己定义一个变量j在每次迭代中存储i的值来解决这个问题,代码如下:
for (var i=0; i<=8; i++) { (function () { var j = i setTimeout(function () { console.log(j) },1000) })(i) }
还有更加简单的解决方案,可以通过let声明来劫持块作用域,并且在这个块作用域中声明一个变量,代码如下:
for (let i=0; i<=8; i++) { setTimeout(function () { console.log(i) },1000) }
for循环头部的let声明会有一个特殊行为,这个行为指出变量在循环的过程中不止被声明一次,每次迭代都会声明,随后每次迭代都会使用上一个迭代结束时的值来初始化这个变量,这就意味着每次循环都会声明一个新的i,且会将上一个迭代结束时的值赋给i
模块模式与闭包模块模式实例:
function module() { //封闭函数 var arr = [1,2,3] function one() { console.log(arr.join("")) } function another() { arr.push(4) console.log(arr.join("")) } return { one: one, //内部函数 another: another } } var accept = module() accept.one() //123 创建一个新的模块实例 accept.another() //1234 创建一个新的模块实例
解析:module是一个函数,需要通过调用它来创建一个模块实例。其返回一个{key:value}形式的对象,这个对象中包含内部函数的引用,将其暴露提供调用,且module内部数据变量是私有隐藏状态,这个对象返回值本质上是这个模块的公共API
分析得出模块模式需要两个必备条件:
必须有外部的封闭函数,该函数必须至少被调用一次(每次调用都会创建一个新的模块实例)
封闭函数必须返回至少一个内部函数,这样内部函数才能在私有作用域中形成闭包,并且可以访问或者修改私有的状态
模块机制var MyModules = (function Manager() { var modules = {}; function define(name, deps, impl) { for (var i=0; i"one"和"another"两个模块都是通过公共API函数来定义,并且"another"接受"one"的实例作为参数注入,这就是模块之间可以存在依赖关系
以上内容是个人的一点总结,如果有错误或不严谨的地方,欢迎批评指正,如果喜欢,欢迎点赞收藏
文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。
转载请注明本文地址:https://www.ucloud.cn/yun/88624.html
摘要:大名鼎鼎的闭包面试必问。闭包的作用是什么。看到闭包在哪了吗闭包到底是什么五年前,我也被这个问题困扰,于是去搜了并总结下来。关于闭包的谣言闭包会造成内存泄露错。闭包里面的变量明明就是我们需要的变量,凭什么说是内存泄露这个谣言是如何来的因为。 本文为饥人谷讲师方方原创文章,首发于 前端学习指南。 大名鼎鼎的闭包!面试必问。请用自己的话简述 什么是「闭包」。 「闭包」的作用是什么。 首先...
摘要:完美的闭包,对,闭包就这么简单。这仅仅是闭包的一部分,闭包利用函数作用域达到了访问外层变量的目的。此时一个完整的闭包实现了,的垃圾回收机制由于闭包的存在无法销毁变量。 1.闭包是指有权访问另一个函数作用域中的变量的函数。 上面这段话来自 javascript 高级程序设计 第三版 P178 。作者说闭包是一个函数,它有访问另一个函数作用域中的变量的能力。 2.函数访问它被创建时所处的...
摘要:到底什么是闭包这个问题在面试是时候经常都会被问,很多小白一听就懵逼了,不知道如何回答好。上面这么说闭包是一种特殊的对象。闭包的注意事项通常,函数的作用域及其所有变量都会在函数执行结束后被销毁。从而使用闭包模块化代码,减少全局变量的污染。 闭包,有人说它是一种设计理念,有人说所有的函数都是闭包。到底什么是闭包?这个问题在面试是时候经常都会被问,很多小白一听就懵逼了,不知道如何回答好。这个...
阅读 1273·2021-11-04 16:09
阅读 3336·2021-10-19 11:45
阅读 2362·2021-10-11 10:59
阅读 984·2021-09-23 11:21
阅读 2743·2021-09-22 10:54
阅读 1107·2019-08-30 15:53
阅读 2577·2019-08-30 15:53
阅读 3454·2019-08-30 12:57