讲一讲闭包

闭包的定义

MDN 对闭包的定义为:闭包就是那些能够访问自由变量的函数。

什么是自由变量呢?

自由变量是指在函数中使用,但既不是函数参数也不是函数局部变量的变量。

由此可见闭包包含两部分:函数+函数能够访问的自由变量,从技术的角度讲,所有的JavaScript函数都是闭包

javascript 复制代码
var a = 1;

function foo() {
    console.log(a);
}

foo();

但这只是理论上的闭包,还有实践角度上的闭包:

ECMAScript中,闭包指的是:

  1. 从理论角度:所有的函数。因为它们都在创建的时候就将上层上下文的数据保存起来了。哪怕是简单的全局变量也是如此,因为函数中访问全局变量就相当于是在访问自由变量,这个时候使用最外层的作用域;
  2. 从实践角度:以下函数才算是闭包:
    1. 即使创建它的上下文已经销毁,它仍然存在(比如,内部函数从父函数中返回);
    2. 在代码中引用了自由变量;

这就涉及到两个知识点:执行上下文和变量对象

对于每个执行上下文,都有三个重要属性:

  • 变量对象(Variable object,VO)
  • 作用域链(Scope chain)
  • this

举一个简单的例子:

javascript 复制代码
var scope = "global scope"
function checkscope(){
	var scope = "local scope"
	function f(){
		console.log(scope)
	}
	return f
}
checkscope()()

执行过程:

  1. 创建全局执行上下文,全局执行上下文压入执行上下文栈,全局执行上下文初始化,初始化的同时,checkscope 函数被创建,保存作用域链到函数的内部属性[[scope]];
javascript 复制代码
ECStack = [
	globalContext
]
globalContext = {
	VO: [global],
	Scope: [globalContext.VO],
	this: globalContext.VO
}
checkscopeContext: {
	scope: [globalContext.VO]
}
  1. 执行checkscope函数,创建checkscope 执行上下文,被压入执行上下文栈;
  2. checkscope函数执行上下文初始化,创建变量对象、作用域链、this等,同时 f 函数被创建,保存作用域链到 f 函数的内部属性[[scope]];
javascript 复制代码
ECStack = [
	checkscopeContext,
	globalContext
]
checkscopeContext: {
	AO: {
		arguments: {
			length: 0
		},
		scope: undefined,
		f: reference to function f(){}
	},
	Scope: [AO, globalContext.VO],
	this: undefined
}
  1. checkscope函数执行完毕,从执行上下文栈弹出;
javascript 复制代码
ECStack = [
	globalContext
]
  1. 执行 f 函数,创建 f 执行上下文,被压入执行上下文栈;
javascript 复制代码
ECStack = [
	fContext,
	globalContext
]
fContext = {
	AO: {
		arguments: {
			length: 0
		},
	},
	Scope: [AO, checkscopeContext.AO, globalContext.VO],
	this: undefined
}
  1. f 函数执行完毕,从执行上下文栈弹出;
  2. 全局执行上下文从执行上下文栈弹出。

f 函数在执行的时候, f 执行上下文维护了一个作用域链,

javascript 复制代码
fContext = {
    Scope: [AO, checkscopeContext.AO, globalContext.VO],
}

因为这个作用域链,f 函数依然可以读取到 checkscopeContext.AO 的值,说明当 f 函数引用了 checkscopeContext.AO 中的值的时候,即使 checkscopeContext 被销毁了,但是 JavaScript 依然会让 checkscopeContext.AO 活在内存中,f 函数依然可以通过 f 函数的作用域链找到它,正是因为 JavaScript 做到了这一点,从而实现了闭包这个概念。

写一个简单的闭包

javascript 复制代码
var arr = []
for(var i = 0; i < 3; i++){
	arr[i] = function(i){
		return function(){
			console.log(i)
		}
	}(i)
}
arr[0]()
arr[1]()
arr[2]()
相关推荐
烛阴7 分钟前
Python装饰器解除:如何让被装饰的函数重获自由?
前端·python
Boilermaker199219 分钟前
【Java EE】Mybatis-Plus
java·开发语言·java-ee
千鼎数字孪生-可视化21 分钟前
Web技术栈重塑HMI开发:HTML5+WebGL的轻量化实践路径
前端·html5·webgl
凌辰揽月21 分钟前
7月10号总结 (1)
前端·css·css3
aramae25 分钟前
C++ -- STL -- vector
开发语言·c++·笔记·后端·visual studio
Tony小周25 分钟前
实现一个点击输入框可以弹出的数字软键盘控件 qt 5.12
开发语言·数据库·qt
天天扭码1 小时前
很全面的前端面试——CSS篇(上)
前端·css·面试
lixzest1 小时前
C++ Lambda 表达式详解
服务器·开发语言·c++·算法
EndingCoder1 小时前
搜索算法在前端的实践
前端·算法·性能优化·状态模式·搜索算法
sunbyte1 小时前
50天50个小项目 (Vue3 + Tailwindcss V4) ✨ | DoubleVerticalSlider(双垂直滑块)
前端·javascript·css·vue.js·vue