JavaScript 中的闭包的形成及使用场景

JavaScript 中的闭包

闭包(Closure) 是 JavaScript 中一个非常重要且独特的概念,它指的是 函数能够记住并访问其词法作用域内的变量,即使这个函数在其词法作用域之外执行。

通俗地说,闭包是 一个函数可以“记住”它在定义时的环境(包括变量和参数),即使这个函数在它定义时的作用域之外被调用,它仍然可以访问该作用域中的变量。

闭包的形成

当一个函数内部定义并返回了另一个函数时,并且这个内部函数引用了外部函数的变量,这就形成了闭包。闭包使得内部函数能够“记住”它定义时外部函数的变量,即使外部函数已经执行结束了,这些变量仍然可以被访问。

闭包的例子
function outer() {let counter = 0;return function inner() {counter++;console.log(counter);}
}const increment = outer();
increment(); // 输出: 1
increment(); // 输出: 2
increment(); // 输出: 3

在这个例子中:

  • outer 函数返回了 inner 函数,而 inner 函数引用了 outer 函数中的局部变量 counter
  • 即使 outer 函数已经执行完毕,inner 函数仍然能够访问和修改 counter,因为 inner 函数形成了一个闭包,保存了对 outer 函数作用域中变量的引用。

闭包的特性

  1. 函数可以记住它的词法作用域:即使外部函数已经返回,闭包中的函数仍然可以访问外部函数作用域中的变量。

  2. 局部变量的持久化:闭包中的局部变量不会在外部函数返回后被销毁,它们会被保留,直到没有任何闭包再引用它们为止。

  3. 数据封装:闭包可以模拟私有变量,避免外部直接访问或修改这些变量,只能通过闭包函数来操作它们。

闭包的使用场景

闭包的应用场景非常广泛,以下是一些常见的使用场景:

1. 模拟私有变量

JavaScript 中没有真正的私有变量,但闭包可以帮助实现类似的效果。通过闭包,可以创建只能通过特定函数访问和修改的变量,避免外部直接访问。

function createCounter() {let count = 0;return {increment: function() {count++;console.log(count);},decrement: function() {count--;console.log(count);}}
}const counter = createCounter();
counter.increment(); // 输出: 1
counter.increment(); // 输出: 2
counter.decrement(); // 输出: 1

在这个例子中,countcreateCounter 函数中的局部变量,外部无法直接访问它,只能通过 incrementdecrement 方法修改它的值。这种方式模拟了私有变量的行为。

2. 函数工厂

闭包可以用于创建返回不同功能的函数,类似于工厂模式。例如,可以创建多个配置不同的计数器:

function createCounter(start) {let count = start;return function() {count++;console.log(count);}
}const counter1 = createCounter(0);
const counter2 = createCounter(100);counter1(); // 输出: 1
counter1(); // 输出: 2
counter2(); // 输出: 101
counter2(); // 输出: 102

每次调用 createCounter 都会生成一个新的计数器实例,且每个计数器都能记住自己的初始值。

3. 回调函数和事件处理

闭包广泛应用于回调函数、事件处理等异步编程场景中。例如,在事件处理函数中,闭包可以保存外部函数的状态,并在事件触发时访问这些状态。

function setupEventListeners() {let name = 'Button clicked!';document.getElementById('myButton').addEventListener('click', function() {console.log(name); // 即使外部函数已执行完毕,仍然能访问 name});
}setupEventListeners();

即使 setupEventListeners 函数执行完毕后,点击按钮时仍然可以访问 name 变量,这是因为 addEventListener 的回调函数形成了闭包。

4. 防抖与节流

闭包常用于防抖和节流函数的实现,以保存计时器状态或限制函数执行频率。例如,防抖函数在用户停止输入一段时间后才执行,可以避免多次频繁的操作。

function debounce(fn, delay) {let timer = null;return function(...args) {clearTimeout(timer);timer = setTimeout(() => {fn.apply(this, args);}, delay);}
}const handleInput = debounce(function(event) {console.log('Input event triggered', event.target.value);
}, 500);document.getElementById('searchInput').addEventListener('input', handleInput);

在这个例子中,闭包用于保存 timer 变量的状态,使得每次用户输入时,timer 不会被销毁,从而实现防抖效果。

5. 缓存(记忆化)

闭包还可以用于缓存计算结果,提高函数的效率。例如,记忆化函数可以缓存已经计算过的结果,避免重复计算。

function memoize(fn) {const cache = {};return function(...args) {const key = JSON.stringify(args);if (cache[key]) {return cache[key];}const result = fn.apply(this, args);cache[key] = result;return result;}
}function slowFunction(num) {console.log('Expensive calculation...');return num * 2;
}const memoizedFunction = memoize(slowFunction);
console.log(memoizedFunction(5)); // 输出: Expensive calculation... 10
console.log(memoizedFunction(5)); // 输出: 10 (缓存结果,没有再计算)

在这里,闭包用于保存计算结果的 cache,当相同的参数再次传递时,直接返回缓存的结果。

闭包的优缺点

优点:
  1. 数据封装:闭包可以帮助创建私有变量,使得某些数据只能通过特定的函数进行访问和修改。
  2. 保持状态:闭包可以保持函数执行时的环境,使得外部函数返回后,依然能够访问和操作这些状态。
  3. 减少全局变量的使用:闭包可以避免使用全局变量,减少变量污染和命名冲突。
缺点:
  1. 内存占用:由于闭包持有外部函数作用域的引用,可能会导致内存得不到及时释放,增加内存消耗,甚至可能引发内存泄漏。
  2. 调试难度:闭包可能导致调试更加复杂,因为内部函数依赖外部的上下文环境,容易引发不可预料的行为。

总结

闭包是 JavaScript 中的一个强大特性,它允许函数在词法作用域之外保持对外部变量的引用。它常用于数据封装回调函数工厂函数防抖/节流等场景。虽然闭包非常强大,但在使用时也要注意内存管理,避免不必要的内存占用和泄漏。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/54955.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

安防区域保护:无线电干扰设备技术详解

在安防区域保护中,无线电干扰设备技术扮演着重要角色,它主要用于通过发射特定频率的无线电波来干扰无人机或其他无线电设备的通信链路、导航信号或控制信号,以达到削弱、阻断甚至控制这些设备运行的目的。以下是对无线电干扰设备技术的详细解…

【GAN】生成对抗网络Generative Adversarial Networks理解摘要

【Pytorch】生成对抗网络实战_pytorch生成对抗网络-CSDN博客 【损失函数】KL散度与交叉熵理解-CSDN博客 [1406.2661] Generative Adversarial Networks (arxiv.org) GAN本质是对抗或者说竞争,通过生成器和鉴别器的竞争获取有效地结果,换句话说&#xff0…

每日一练 2024.9.29(2)

目录 解题思路与代码实现 题目分析 一、解题策略 关键步骤: 二、代码实现 三、代码解析 四、复杂度分析 五、运行示例 示例1: 示例2: 六、总结 解题思路与代码实现 题目分析 这道题目要求我们找到字符串列表 strs 中的相似字符组…

C++——vector

1.简介 2.成员函数 2.1构造函数 void test_vector1() {//1.无参构造vector<int> v1;cout << v1.capacity() << endl;//2.传参构造vector<int> v2(10,1);//3.迭代器构造vector<int> v3(v2.begin(), v2.end());//也可以使用其它容器的迭代器区间来…

scrapy快速上手

安装 除了scrapy本身还要安装两个库 pip install scrapy pip install pywin32 pip install wheel 创建项目 在要创建项目的地方打开powershell scrapy startproject 项目名 我们得到这样的项目结构&#xff0c;功能如下 scrapy.cfg 项目的主配置信息 …

LeetCode[中等] 17. 电话号码的字母组合

给定一个仅包含数字 2-9 的字符串&#xff0c;返回所有它能表示的字母组合。答案可以按 任意顺序 返回。 给出数字到字母的映射如下&#xff08;与电话按键相同&#xff09;。注意 1 不对应任何字母。 思路 回溯法 log&#xff1a;当前结果数组&#xff1b;level&#xff1a…

第五届计算机科学与管理科技国际学术会议(ICCSMT 2024)

梁哲&#xff0c;同济大学长聘特聘教授&#xff0c;国家杰青、首届国家杰青延续项目获得者、上海市曙光学者、上海市优秀学术带头人。本科毕业于新加坡国立大计算机工程系、硕士毕业于新加坡国立大学工业与系统工程系、博士毕业于美国新泽西州立大学工业工程系。理论研究主要集…

修改Opcenter EXFN 页面超时时间(Adjust UI Session Extend Token)

如果你想修改Opcenter EXFN中页面Session的超时时间&#xff0c;你可以按照如下步骤修改SessionAge 这个参数&#xff1a; 管理员运行CMD执行以下命令 umconf -getconfig -file C:\temp\config.json如果第2步有报错&#xff0c;则执行步骤4;如果没有报错则执行第5步如果第2步…

探索光耦:光耦在电脑电源中的应用及其重要性

随着计算机技术的飞速发展&#xff0c;电脑已成为现代生活和工作中不可或缺的工具。无论是日常办公、游戏娱乐还是复杂的图像处理&#xff0c;电脑电源的稳定性和安全性都至关重要。作为电脑电源的核心部件之一&#xff0c;光耦&#xff08;光电耦合器&#xff09;在提升电源性…

JavaScript网页设计案例:互动式简历网站

JavaScript网页设计案例&#xff1a;互动式简历网站 在现代网页设计中&#xff0c;JavaScript 是实现交互和动态效果的关键技术。本文将通过一个完整的案例&#xff0c;展示如何使用 JavaScript 构建一个交互式的个人简历网页。本文不仅会涵盖 HTML 和 CSS 的使用&#xff0c;…

android和ios双端应用性能的测试工具

1.工具介绍 基于日常工作的需要&#xff0c;开发了一款新的android和ios端应用性能测试工具&#xff0c;本工具在数据测试方面与所流行的工具没有区别。欢迎下载使用体验。 本工具为筋斗云&#xff0c;工具说明 本工具无侵入&#xff0c;不需要root&#xff0c;低延迟…

(十七)、Mac 安装k8s

文章目录 1、Enable Kubernetes2、查看k8s运行状态3、启用 kubernetes-dashboard3.1、如果启动成功&#xff0c;可以在浏览器访问3.2、如果没有跳转&#xff0c;需要单独安装 kubernetes-dashboard3.2.1、方式一&#xff1a;一步到位3.2.2、方式二&#xff1a;逐步进行 1、Enab…

如何恢复被删除的 GitLab 项目?

GitLab 是一个全球知名的一体化 DevOps 平台&#xff0c;很多人都通过私有化部署 GitLab 来进行源代码托管。极狐GitLab 是 GitLab 在中国的发行版&#xff0c;专门为中国程序员服务。可以一键式部署极狐GitLab。 学习极狐GitLab 的相关资料&#xff1a; 极狐GitLab 官网极狐…

time命令:轻松测量Linux命令执行时间!

一、命令简介 用途&#xff1a; 用于测量 Linux 命令执行的时间&#xff0c;包括实际时间、用户 CPU 时间和系统 CPU 时间。刚开始以为是用来“看现在几点钟”的 &#x1f972;。标签&#xff1a; 实用工具&#xff0c;性能分析。 ‍ 二、命令参数 2.1 命令格式 time [选项…

进程的那些事--实现shell

目录 前言 一、预备知识 二、实现步骤 1.思路 2.实现 总结 前言 提示&#xff1a;这里可以添加本文要记录的大概内容&#xff1a; 学习的本质就是变现 提示&#xff1a;以下是本篇文章正文内容&#xff0c;下面案例可供参考 一、预备知识 char * fgets ( char * str, i…

讯飞星火编排创建智能体学习(二)决策节点

目录 概述 决策节点 文生图节点 连接节点 测试结果 概述 在上一篇博文讯飞星火编排创建智能体学习&#xff08;一&#xff09;最简单的智能体构建-CSDN博客&#xff0c;我介绍了编排创作智能体&#xff0c;这篇来介绍一下“决策节点”。 决策节点 在编排创作智能体中&…

MQTT.fx 1.7.1使用说明篇(OneNET-MQTT-API调试)

&#xff08;代码完美实现&#xff09;stm32 新版 onenet mqtt物联网(保姆级教程) &#xff08;代码完美实现&#xff09;stm32 新版 onenet mqtt物联网(保姆级教程)https://blog.csdn.net/Wang2869902214/article/details/142501323 MQTT.fx 1.7.1使用教程 下载地址 MQ…

巧用switch-case消除条件判断

shigen坚持更新文章的博客写手&#xff0c;记录成长&#xff0c;分享认知&#xff0c;留住感动。个人IP&#xff1a;shigen 在之前的文章中&#xff0c;我们有提交消除if-else代码的方法&#xff1a; 结合HashMap与Java 8的Function和Optional消除ifelse判断巧用枚举消除逻辑判…

一文上手SpringSecuirty【六】

自定义认证流程完成之后,前端收到了后端生成的token,那么在之后的所有请求当前,都必须携带token.作为服务器来说,得验证这个token,是否合法. 一、验证token是否合法 1.1 OncePerRequestFilter过滤器 OncePerRequestFilter是 Spring 框架中的一个过滤器&#xff0c;用于确保在…

抖音支付回调验签 go 版本

序言 最近在做抖音小程序支付&#xff0c;由于抖音开放平台的文档写的较为简陋&#xff0c;让人踩了不少坑&#xff0c;在这里整理一下做小程序支付的整个过程&#xff0c;以通用交易系统为例子。 准备条件 1&#xff09;申请小程序&#xff0c;开通支付功能 这里需要明确你小…