网站首页 > 技术文章 正文
前置知识:
声明函数两种方法:
- 函数声明,存在函数声明提升,因此可以在函数声明之前调用(不会报错)。
- 函数表达式,不存在函数声明提升,若定义前调用,会报错(函数还不存在)。
1.概念
1.1 词法作用域
这里先要了解一个概念,词法作用域:它是静态的作用域,是书写变量和块作用域的作用域。
由于函数g的作用域中没有a这个变量,但是它可以访问父作用域,并使用父作用域下的变量a,最后输出"leo"。
词法作用域中使用的域,是变量在代码中声明的位置所决定的。嵌套的函数可以访问在其外部声明的变量。
1.2 闭包
接下来介绍下闭包概念,闭包是指有权访问另一个函数作用域中的变量的函数。
闭包是由函数以及创建该函数的词法环境组合而成。这个环境包含了这个闭包创建时所能访问的所有局部变量。
创建闭包的常见方式:在一个函数内创建另一个函数。如:
通过概念可以看出,闭包有以下三个特征:
- 函数嵌套函数
- 函数内部可以引用函数外部的参数和变量
- 参数和变量不会被垃圾回收机制回收
注:关于内存回收机制,可以查看阮一峰老师的《JavaScript 内存泄漏教程》。
另外,使用闭包有以下好处:
- 将一个变量长期保存在内存中
- 避免全局变量的污染
因为垃圾回收机制没有回收,所以每次调用fun()都会返回新的值。
- 私有化成员,使得外部不能访问
2.易错点
2.1 引用的变量发生变化
原本照我们的想法,fun方法中每个元素上的方法执行的结果应该是1,2,3,...,10,而实际上,每个返回都是10,因为每个闭包函数引用的变量i是f执行环境下的变量i,循环结束后,i已经变成10,所以都会返回10。
解决办法可以这样:
2.2 this指向问题
由于里面的闭包函数是在window作用域下执行,因此this指向window。
2.3 内存泄漏
当我们在闭包内引用父作用域的变量,会使得变量无法被回收。
这样做的话,变量a会一直存在无法释放,类似的变量越来越多的话,很容易引起内存泄漏。我们可以这么解决:
通过把变量赋值成null来主动释放掉。
3.案例
3.1 经典案例——定时器和闭包
代码如下:
不出所料,返回的不是我们想要的0,1,2,3,...,9,而是10个10。
这是因为js是单进程,所以在执行for循环的时候定时器setTimeout被安排到任务队列中排队等候执行,而在等待过程中,for循环已经在执行,等到setTimeout要执行的时候,for循环已经执行完成,i的值就是10,所以就打印了10个10。
解决方法 :
- 1.使用ES6新增的let。
把for循环中的var替换成let。
- 2.使用闭包
3.2 使用闭包解决递归调用问题
因为最好是return num* arguments.callee(num-1),arguments.callee指向当前执行函数,但是在严格模式下不能使用该属性也会报错,所以借助闭包来实现
这里可以使用return num >1 ? num* arguments.callee(num-1) : 1;,因为arguments.callee指向当前执行函数,但是在严格模式下不能使用,也会报错,所以这里需要使用闭包来实现。
这样做,实际上起作用的是闭包函数f,而不是外面的fun。
3.3 使用闭包模仿块级作用域
ES6之前,使用var声明变量会有变量提升问题:
为了避免这个问题,我们这样使用闭包(匿名自执行函数):
我们创建了一个匿名的函数,并立即执行它,由于外部无法引用它内部的变量,因此在函数执行完后会立刻释放资源,关键是不污染全局对象。这里i随着闭包函数的结束,执行环境销毁,变量回收。
但是现在,我们用的更多的是ES6规范的let和const来声明。
公众号:前端自习课
往期回顾
「JavaScript 从入门到精通」2.流程控制和错误处理
猜你喜欢
- 2025-06-23 JavaScript 开发者常犯的 10 个典型错误
- 2025-06-23 这7道问题可以检验你掌握Javascript的真正实力
- 2025-06-23 HarmonyOS NEXT 如何有效地进行内存管理和避免内存泄露?
- 2025-06-23 JS 函数柯里化(js函数柯里化面试题)
- 2025-06-23 《深入理解javascript原型和闭包系列》 知识点整理
- 2025-06-23 JS面试题突破:什么是闭包?(闭包 js)
- 2025-06-23 JavaScript闭包(javascript 闭包)
- 2025-06-23 “闭包到底是什么?90% 的前端开发者都没真正搞懂!”
- 2024-10-04 两个小案例让你深入理解 Javascript 闭包和作用域
- 2024-10-04 js中的闭包 js闭包的使用
你 发表评论:
欢迎- 561℃Oracle分析函数之Lag和Lead()使用
- 556℃几个Oracle空值处理函数 oracle处理null值的函数
- 546℃Oracle数据库的单、多行函数 oracle执行多个sql语句
- 542℃0497-如何将Kerberos的CDH6.1从Oracle JDK 1.8迁移至OpenJDK 1.8
- 539℃Oracle 12c PDB迁移(一) oracle迁移到oceanbase
- 531℃【数据统计分析】详解Oracle分组函数之CUBE
- 519℃最佳实践 | 提效 47 倍,制造业生产 Oracle 迁移替换
- 510℃Oracle有哪些常见的函数? oracle中常用的函数
- 最近发表
- 标签列表
-
- 前端设计模式 (75)
- 前端性能优化 (51)
- 前端模板 (66)
- 前端跨域 (52)
- 前端缓存 (63)
- 前端react (48)
- 前端aes加密 (58)
- 前端脚手架 (56)
- 前端md5加密 (54)
- 前端富文本编辑器 (47)
- 前端路由 (61)
- 前端数组 (73)
- 前端排序 (47)
- 前端密码加密 (47)
- Oracle RAC (73)
- oracle恢复 (76)
- oracle 删除表 (48)
- oracle 用户名 (74)
- oracle 工具 (55)
- oracle 内存 (50)
- oracle 导出表 (57)
- oracle 中文 (51)
- oracle的函数 (57)
- 前端调试 (52)
- 前端登录页面 (48)
本文暂时没有评论,来添加一个吧(●'◡'●)