思维导图
闭包
上级作用域的概念
- 函数的上级作用域在哪里创建创建的,上级作用域就是谁
var a = 10
function foo(){
console.log(a)
}
function sum() {
var a = 20
foo()
}
sum()
/* 输出
10
/
复制代码
思考题
var n = 10
function fn(){
var n =20
function f() {
n++;
console.log(n)
}
f()
return f
}
var x = fn()
x()
x()
console.log(n)
/* 输出
* 21
22
23
10
/
复制代码
- 关于如何查找上级作用域
JS 堆栈内存释放
堆内存:存储引用类型值,对象类型就是键值对,函数就是代码字符串。
堆内存释放:将引用类型的空间地址变量赋值成
null
,或没有变量占用堆内存了浏览器就会释放掉这个地址栈内存:提供代码执行的环境和存储基本类型值。
栈内存释放:一般当函数执行完后函数的私有作用域就会被释放掉。
闭包是什么
形成闭包的原因
var a = 0
function foo(){
var b =14
function fo(){
console.log(a, b)
}
fo()
}
foo()
复制代码
闭包的作用
- 保护函数的私有变量不受外部的干扰。形成不销毁的栈内存。
- 保存,把一些函数内的值保存下来。闭包可以实现方法和属性的私有化
闭包经典使用场景
-
return
回一个函数
-
var n = 10
function fn(){
var n =20
function f() {
n++;
console.log(n)
}
return f
}
var x = fn()
x() // 21
复制代码
- 函数作为参数
var a = '林一一'
function foo(){
var a = 'foo'
function fo(){
console.log(a)
}
return fo
}
function f(p){
var a = 'f'
p()
}
f(foo())
/* 输出
* foo
/
复制代码
- IIFE(自执行函数)
var n = '林一一';
(function p(){
console.log(n)
})()
/* 输出
* 林一一
/
复制代码
- 循环赋值
for(var i = 0; i<10; i++){
(function(j){
setTimeout(function(){
console.log(j)
}, 1000)
})(i)
}
复制代码
- 使用回调函数就是在使用闭包
window.name = '林一一'
setTimeout(function timeHandler(){
console.log(window.name);
}, 100)
复制代码
使用闭包需要注意什么
经典面试题
- for 循环和闭包(号称必刷题)
var data = [];
for (var i = 0; i < 3; i++) {
data[i] = function () {
console.log(i);
};
}
data[0]();
data[1]();
data[2]()
/* 输出
3
3
3
/
复制代码
- 使用闭包改善上面的写法达到预期效果,写法1:自执行函数和闭包
var data = [];
for (var i = 0; i < 3; i++) {
(function(j){
setTimeout( data[j] = function () {
console.log(j);
}, 0)
})(i)
}
data[0]();
data[1]();
data[2]()
复制代码
- 写法2:使用
let
var data = [];
for (let i = 0; i < 3; i++) {
data[i] = function () {
console.log(i);
};
}
data[0]();
data[1]();
data[2]()
复制代码