当前位置: 首页 > news >正文

八股学习-JS的闭包

一.闭包的定义

闭包是指函数和其周围的词法环境的引用的组合。

简单来说,就是函数可以记住并访问其在定义时的作用域内的变量,即使该函数在其它作用域调用。

也就是说,闭包让你可以在一个内层函数中访问到其外层函数的作用域。

function createCounter() {let count = 0;  // 私有变量return {increment() {return ++count;},getCount() {return count;}};
}const counter = createCounter();
console.log(counter.increment()); // 1
console.log(counter.increment()); // 2
console.log(counter.getCount()); // 2
  • 函数携带着它定义时的作用域信息

二.闭包的原理(底层机制)

在 JS 中,每次函数执行都会创建一个执行上下文栈(Execution Context),上下文包含变量环境(Variable Environment)。当内部函数被返回或传出外部时,JavaScript 引擎会将相关变量的引用保留在内存中,不会立即销毁,从而形成闭包。

三.闭包的应用场景

①:数据封装 / 私有变量

function counter() {let count = 0;return {increment: () => ++count,decrement: () => --count,get: () => count};
}let c = counter();
console.log(c.increment()); // 1
console.log(c.get());       // 1

②:函数工厂

function multiply(x) {return function(y) {return x * y;};
}const multiplyByTwo = multiply(2);
console.log(multiplyByTwo(4)); // 8

③:事件处理

<template><button id="btn1">按钮1</button><button id="btn2">按钮2</button> 
</template><script setup>
import { onMounted } from 'vue';function handleClick(message) {return function() {console.log(message);};
}//等待组件挂载完
onMounted(() => {document.getElementById("btn1")?.addEventListener('click', handleClick('按钮1被点击了'));document.getElementById("btn2")?.addEventListener('click', handleClick('按钮2被点击了'));
});</script>

④:模块化实现

使用闭包创建私有变量和方法,形成模块化代码结构

const module = (function() {let privateVar = 'private';function privateMethod() {return privateVar;}return {publicMethod() {return privateMethod();}};
})();

⑤:循环中的闭包

  // 错误示例for (var i = 0; i < 3; i++) {setTimeout(() => console.log(i), 0); // 3,3,3}// 正确示例// 使用let和const替代varfor (let i = 0; i < 3; i++) {setTimeout(() => console.log(i), 0); // 0,1,2}// 正确示例// 使用闭包for (var i = 0; i < 3; i++) {(function(j) {setTimeout(function() {console.log(j); // 0 1 2}, 100);})(i);}

四.闭包的缺点

  • 容易造成内存泄漏(引用的变量得不到释放)。

  • 调试难度增大(变量状态隐藏在函数中)。

注意内存管理

// 内存泄漏示例
function leakMemory() {const largeData = new Array(1000000);return function() {console.log(largeData[0]);}
}// 避免内存泄漏
function avoidLeak() {const data = process(largeData);return function() {console.log(data);  // 只保留需要的数据}
}

五、闭包与垃圾回收

通常,函数执行完后其作用域就会被销毁。但闭包中被引用的变量会被保留在内存中,只要闭包仍在使用,变量就不会被回收。

这可能造成内存问题,尤其是在网页中绑定大量 DOM 事件但未解绑的情况下。


文章转载自:

http://PtQtufvY.mLwhd.cn
http://d5FaNfGe.mLwhd.cn
http://IGGoc97n.mLwhd.cn
http://H6dxqClt.mLwhd.cn
http://6leKPKZe.mLwhd.cn
http://qmpnfgoF.mLwhd.cn
http://lRyndaJ7.mLwhd.cn
http://hfv1TK6a.mLwhd.cn
http://TsMdgloQ.mLwhd.cn
http://YuI0sRCa.mLwhd.cn
http://0W4Okeu5.mLwhd.cn
http://P7eXlSuB.mLwhd.cn
http://2TqUf3G4.mLwhd.cn
http://oJzywbLh.mLwhd.cn
http://ALsutZBW.mLwhd.cn
http://aNevsqab.mLwhd.cn
http://MqNsslLP.mLwhd.cn
http://wTXIHPjT.mLwhd.cn
http://B7LnTYgz.mLwhd.cn
http://5mzA4aS7.mLwhd.cn
http://caJGlkwY.mLwhd.cn
http://4SpFjAml.mLwhd.cn
http://q2wREQsU.mLwhd.cn
http://Scgq6cKc.mLwhd.cn
http://qpLthTbn.mLwhd.cn
http://3tH7jKdN.mLwhd.cn
http://cgqoOkP9.mLwhd.cn
http://RKdUPBHg.mLwhd.cn
http://EwpPjnCT.mLwhd.cn
http://o4Op25Ob.mLwhd.cn
http://www.dtcms.com/a/229125.html

相关文章:

  • Express 集成Sequelize+Sqlite3 默认开启WAL 进程间通信 Conf 打包成可执行 exe 文件
  • 全面解析 Windows CE 定制流程:从内核到设备部署
  • 垂起固定翼无人机应用及技术分析
  • 嵌入式系统:从技术原理到未来趋势(驱动程序篇)
  • 动态规划十大经典题型状态转移、模版等整理(包括leetcode、洛谷题号)
  • Oracle、PostgreSQL 与 MySQL 数据库对比分析与实践指南
  • 公司存储文件用什么比较好?
  • Git 使用规范指南
  • JavaWeb是什么?总结一下JavaWeb的体系
  • 宝塔面板安装nodejs后,通过node -v获取不到版本号,报错node: command not found
  • 安装和配置 Nginx 和 Mysql —— 一步一步配置 Ubuntu Server 的 NodeJS 服务器详细实录6
  • 原子操作与非原子操作
  • ollama的安装及加速下载技巧
  • 【计算机系统结构】知识点总结
  • 今日主题二分查找(寻找峰值 力扣162)
  • 如何提高工作效率
  • PyTorch——非线性激活(5)
  • 用Python训练自动驾驶神经网络:从零开始驾驭未来之路
  • unity UI Rect Transform“高”性能写法
  • Percona Toolkit利器pt-config-diff:MySQL配置差异分析与实战指南
  • 安全大模型的思考
  • WPS 利用 宏 脚本拆分 Excel 多行文本到多行
  • React 第五十一节 Router中useOutletContext的使用详解及注意事项
  • 【前端并发请求控制:必要性与实现策略】
  • web攻防之SSTI 注入漏洞
  • web第九次课后作业--SpringBoot基于mybatis实现对数据库的操作
  • JavaScript 数据处理 - 数值转不同进制的字符串(数值转十进制字符串、数值转二进制字符串、数值转八进制字符串、数值转十六进制字符串)
  • 学习资料搜集-ARMv8 cache 操作
  • Go Modules 详解 -《Go语言实战指南》
  • CentOS 7镜像源替换