闭包
闭包就是能够读取其他函数内部变量的函数。例如在javascript中,只有函数内部的子函数才能读取局部变量,所以闭包可以理解成“定义在一个函数内部的函数“。在本质上,闭包是将函数内部和函数外部连接起来的桥梁。
接下来看三段代码
|
第一段代码是无法实现我们想要实现的结果的,而后两者可以实现
如何产生闭包?
当一个嵌套的内部(子)函数引用了嵌套的外部(父)函数的变量(函数)时, 就产生了闭包
执行函数定义就会产生闭包(不用调用内部函数)
闭包到底是什么?
- 使用chrome调试查看
function fn1 () {
var a = 2
var b = 'abc'
fn2()
function fn2 () { //执行函数定义就会产生闭包(不用调用内部函数)
console.log(a)
}
}
fn1()
- 理解一: 闭包是嵌套的内部函数(绝大部分人)
- 理解二: 包含被引用变量(函数)的对象(极少数人)
注意: 闭包存在于嵌套的内部函数中
产生闭包的条件?
- 函数嵌套
- 内部函数
引用了外部函数的数据
(变量/函数)
常见的闭包
- 将函数作为另一个函数的返回值
function fn1() {
var a = 2
function fn2() {
a++
console.log(a)
}
return fn2
}
var f = fn1()
f() // 3
f() // 4为了方便理解,此处返回的
fn2
则为图中标注这块儿
- 将函数作为实参传递给另一个函数调用
function showDelay(msg, time) {
setTimeout(function () {
alert(msg)
}, time)
}
showDelay('闭包', 2000)
闭包的作用
作用:
- 延长局部变量的生命周期——缓存数据
- 如果想要缓存数据,就把这个数据放在外层的函数和里层的函数的中间位置
- 局部变量是在函数中,函数使用结束后,局部变量就会被自动的释放
- 闭包后,里面的局部变量的使用作用域链就会被延长
- 让函数外部能操作内部的局部变量
闭包的生命周期
产生
在嵌套内部函数定义执行完时就产生了
(不是在调用)
死亡
在嵌套的内部函数成为垃圾对象时
function fn1() { |
闭包的应用
- 具有特定功能的js文件
- 将所有的数据和功能都封装在一个函数内部(私有的)
- 只向外暴露一个包信n个方法的对象或函数
- 模块的使用者, 只需要通过模块暴露的对象调用方法来实现对应的功能
自定义JS模块
模块1
myModule.js
function myModule() { |
file1
<script type="text/javascript" src="myModule.js"></script> |
模块2
myModule2.js
(function () { // 匿名函数自调用 |
file2
<script type="text/javascript" src="myModule2.js"></script> |
可以发现模块2的好处是不用定义方法,可直接采取调用,使用更加简单方便
闭包的缺点与解决方法
- 缺点
- 函数执行完后, 函数内的局部变量没有释放, 占用内存时间会变长
- 容易造成内存泄露
- 解决
- 能不用闭包就不用
- 及时释放
function fn1() {
var arr = new Array(100000)
function fn2() {
console.log(arr.length)
}
return fn2
}
var f = fn1()
f()
f = null //让内部函数成为垃圾对象-->回收闭包
闭包面试题
题一
//代码片段一 |