一文了解前端面试重点--闭包

1、什么是闭包?

闭包是指有权访问另一个函数作用域中的变量的函数,创建闭包的最常见的方式就是在一个函数内创建另一个函数。

温馨提示:由于闭包所在的作用域返回的局部变量不会被销毁,所以会占用内存。过度的使用闭包会迫使性能下降,因此建议大家在有必要的情况下再使用闭包。

闭包的官方解释:

闭包(closure)是一个函数以及其捆绑的周边环境状态(lexical environment词法环境)的引用的组合。换而言之,闭包让开发者可以从内部函数访问外部函数的作用域。在 JavaScript 中,闭包会随着函数的创建而被同时创建。

function fn(){var arr = [];for(var i = 0; i < 3; i++){arr.push(function(){return i;});}return arr;
}
var b = fn();
for(var i = 0; i < b.length; i++){console.log(b[i]());
}

这个控制台输出的是三个3,你知道为什么嘛?

答:这是因为在第3行的这个for循环执行的过程当中,每次都只是向数组中添加了一个匿名函数,但是你要知道这个时候这些函数并没有自我执行。当你通过b[i]()调用匿名函数的时候,通过闭包获得的i已经是3了,所以每次输出的都是3。 

那么如果你想得到0,1,2怎么办呢?这个时候你就需要知道闭包怎么用了。

方法一:(闭包+立即执行函数表达式)

function fn(){var arr = [];for(var i = 0; i < 3; i++){arr.push((function(num){return function(){return num;};})(i));}return arr;
}
var b = fn();
for(var i = 0; i < b.length; i++){console.log(b[i]());
}

将立即执行函数表达式(IIFE) (function(num) { ... })(i) 包裹在 arr.push() 中,这样可以在每次循环时创建一个新的作用域并传入当前的 i 值作为参数 num。这样,内部的匿名函数就能够捕获并保存每次循环时的 i 值,从而避免了之前出现的当匿名函数执行时i值已经变为3的问题。

通过匿名函数的立即执行,将立即执行后返回的函数直接赋值给数组arr。每次循环即将i的值传递给num,又因为num在函数中,所以有自己的独立作用域,因此num得到的值为每次循环传递进来的i值,即0,1,2 

方法二:(只是把闭包函数单独抽离出来了而已)

function createClosure(num) {return function() {return num;};
}function fn() {var arr = [];for (var i = 0; i < 3; i++) {arr.push(createClosure(i));}return arr;
}var b = fn();
for (var i = 0; i < b.length; i++) {console.log(b[i]());
}

定义了一个 createClosure 函数,它接受一个参数 num 并返回一个闭包函数,该闭包函数可以访问并返回 num 的值。闭包可以帮助我们在每次循环中创建一个独立的作用域,并保留每次循环中变量i的值。

在JavaScript中,当内部函数引用了外部函数的变量时,会创建一个闭包,使得内部函数能够访问和保留外部函数作用域中的变量。这样,每个内部函数都有自己的 num 变量,它们分别对应每次循环中的 i 值,避免了之前版本中由于共享同一个 i 变量导致的问题。

关于使用闭包时的this指向问题:

var color = "red";
function fn(){return this.color;
}
var obj = {color:"yellow",fn:function(){return function(){//返回匿名函数return this.color;}}
}
console.log(fn());//red 在外部直接调用this为window
var b = obj.fn();//b为window下的变量,获得的值为obj对象下的fn方法返回的匿名函数
console.log(b());//red 因为是在window环境下运行,所以this指缶的是window
//可以通过call或apply改变函数内的this指向
console.log(b.call(obj));//yellow
console.log(b.apply(obj));//yellow
console.log(fn.call(obj));//yellow//或者这个保存对当前对象的引用
var color = "red";
function fn(){return this.color;
}
var obj = {color: "yellow",fn: function(){var self = this; // 保存对当前对象的引用return function(){return self.color; // 使用保存的引用来访问对象的color属性}}
}
var b = obj.fn();
console.log(b()); // 输出 "yellow"

总结:

闭包解决了以下两个问题:

1. 可以读取函数内部的变量

2. 让这些变量的值始终保持在内存中。不会在函数调用后被清除

但同时,闭包也有它的缺点:

1. 由于闭包会使得函数中的变量都被保存到内存中,滥用闭包很容易造成内存消耗过大,导致网页性能问题。解决方法是在退出函数之前,将不再使用的局部变量全部删除。

闭包会导致内存泄漏。内存泄漏指的是在程序中,不再需要使用的内存没有被释放或回收的情况。当程序中的对象在不再需要时仍然占用内存空间而没有被垃圾回收机制释放,就会导致内存泄漏问题。造成内存泄漏的情况可能有:定时器未清理,闭包,对象相互引用等。

我们在使用闭包时需要特别注意内存泄漏的问题,可以用以下两种方法解决内存泄露问题:

1.及时释放闭包:手动调用闭包函数,并将其返回值赋值为null,这样可以让闭包中的变量及时被垃圾回收器回收。
2.使用立即执行函数:在创建闭包时,将需要保留的变量传递给一个立即执行函数,并将这些变量作为参数传递给闭包函数,这样可以保留所需的变量,而不会导致其他变量的内存泄漏。

2. 闭包可以使得函数内部的值可以在函数外部进行修改。所以,如果你把父函数当作对象(object)使用,把闭包当作它的公用方法(Public Method),把内部变量当作它的私有属性(private value),这时一定要小心,不要随便改变父函数内部变量的值。

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

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

相关文章

GraphQL入门之使用ApolloServer和express构建GraphQL服务

接上一篇文章&#xff0c;由于 express 现在仍然是主流的 Node.js 服务端框架&#xff0c;所以今天看看 ApolloServer 怎样和 express 集成构建 GraphQL 服务。另外今天文章也顺便讲一下怎么使用 typescript 来实现。 初始化项目 mkdir myapp cd myapp npm init (一路回车)安…

FPGA - 单总线协议(one-wire)

1&#xff0c;简介 单总线&#xff08;one-wire&#xff09;是美国 DALLAS 公司推出的外围串行扩展总线技术&#xff0c;与 SPI、I2C 等串行数据通信方式不同&#xff0c;它采用单根信号线&#xff0c;既传输时钟又传输数据&#xff0c;而且数据传输是双向的。它具有节省 I/O口…

【机器人控制 Robot Control】非线性控制(Non-linear Control)建模举例【新加坡南洋理工大学 NTU Singapore】

Non-linear Control Method Example: Non-linear Mechanical System Modelling of the System using Control Law Partitioning (Handwritten)

Lua 如何在Lua中调用C/C++函数

Lua调用C函数有两种方式 程序主体在C中运行&#xff0c;C函数注册到Lua中。C调用Lua&#xff0c;Lua调用C注册的函数&#xff0c;C或者Lua得到函数的执行结果。程序主体在Lua中运行&#xff0c;C函数作为库函数供Lua使用。 C的代码如下 如何在Lua脚本中调用这个C语言函数(ad…

医药电商大数据实践

文章目录 一、项目概述二、功能概述三、关键技术四、系统架构设计4.1 大数据采集子系统4.1.1定义4.1.2接口五、数据存储设计5.2数据采集子系统5.2.1采集用户流量数据5.2.2采集用户订单数据5.2.3 采集用户订单项数5.3流量分析及经营状况分析5.3.1流量分析5.3.2经营状况分析数据5…

【Ubuntu-20.04】OpenCV-3.4.16的安装并对图片与视频处理

【Ubuntu-20.04】OpenCV-3.4.16的安装并对图片与视频处理 一、安装OpenCV-3.4.161.下载OpenCV-3.4.16安装包2.将安装包放到/home&#xff0c;并解压3.使用 cmake 安装 opencv4.配置环境5.查看 opencv 的版本信息 二、处理图片&#xff08;一&#xff09;创建文件夹 code &#…

Ansible管理主机的清单------------inventory

一、 Ansible组成 INVENTORY&#xff1a;Ansible管理主机的清单 /etc/ansible/hosts 需要管理的服务清单,(将你需要管理的主机 、地址 或者名字 写入此文件) MODULES&#xff1a;Ansible执行命令的功能模块&#xff0c;多数为内置核心模块&#xff0c;也可自定义 PLUGINS&…

观测云在 .NET 业务中分析性能问题的最佳实践

背景 某药业集团是一家以创新技术驱动的线下医疗数据 SaaS 平台建设和运营公司&#xff0c;其主营的某智慧医疗平台产品&#xff0c;围绕线下医疗场景痛点提供一体化服务解决方案。近期集团对其生物检材在线递检系统进行功能升级开发及 IaaS 平台迁移。在针对新系统和新基础设…

vue3/vue2若依框架对比,点击新增编辑跳转到新页面(新增编辑共用代码)

vue2若依框架&#xff1a; router里面定义好&#xff0c;编辑里面添加一个id {path: /filmManagement,component: Layout,hidden: true,redirect: noredirect,children: [{path: editFilmDetail,component: () > import(/views/filmManagement/editFilmDetail),name: editFi…

【分布式websocket】聊天系统消息加密如何做

前言 先介绍一下对称加密算法&#xff0c;在介绍一下加密流程&#xff0c;然后是介绍一下查询加密消息的策略。然后结合现有技术架构然后去选型。 决定采用客户端解密。简而言之就是采用对称服务端加密。然后将加密内容存储到消息表的content字段。然后客户拉取content字段 然…

旅游景区公共广播 园区广播 公路服务区广播

旅游景区公共广播 园区广播 公路服务区广播 旅游景区公共广播 旅游景区公共广播(又称背景音乐)简称BGM&#xff0c;它的主要作用是掩盖噪声并创造一种轻松和谐的气氛&#xff0c;是一种创造轻松愉快环境气氛的音乐。掩盖环境噪声&#xff0c;创造与旅游景区相适应的气氛&#…

报错:Nginx 部署后刷新页面 404 问题

文章目录 问题分析解决 问题 在部署完项目后 刷新页面&#xff0c;页面进入了404 分析 加载单页应用后路由改变均由浏览器处理&#xff0c;而刷新时将会请求当前的链接&#xff0c;而Nginx无法找到对应的页面 关键代码try_files,剩下俩如果其他地方配置了则可以省略。 在这…

GPT实战系列-LangChain构建自定义Agent

GPT实战系列-LangChain构建自定义Agent LangChain GPT实战系列-LangChain如何构建基通义千问的多工具链 GPT实战系列-构建多参数的自定义LangChain工具 GPT实战系列-通过Basetool构建自定义LangChain工具方法 GPT实战系列-一种构建LangChain自定义Tool工具的简单方法 GPT…

ArmSoM Rockchip系列产品 通用教程 之 RTC 使用

1. RTC 简介​ RTC&#xff1a;(Real_Time Clock)&#xff1a;实时时钟 HYM8563是一种低功耗实时时钟&#xff08;RTC&#xff09;芯片&#xff0c;用于提供精确的时间和日期信息。它提供一个可编程的时钟输出&#xff0c;一个中断输出和一个掉电检测器&#xff0c;所有的地址…

新装 mysql 需注意配置的项

一 时区配置 mysql 安装后默认是使用系统时间&#xff0c;如果是linux系统就要注意系统时区是否配置正确 1 配置centos系统时区 # 查看当前系统时区 timedatectl status# 如果需要更改时区&#xff0c;比如设置为北京时间&#xff08;中国标准时间&#xff09; sudo timeda…

柔性数组知识点

什么是柔性数组 什么是柔性数组&#xff1f;柔性数组其实也是动态内存管理部分的内容。这节主要来复习柔性数组的知识点。 当结构体的最后一个成员为数组&#xff0c; 且这个数组的大小未确定时&#xff0c; 我们就称它是柔性数组。 如&#xff1a; struct stu{char name[…

【Python使用】嘿马头条完整开发md笔记第1篇:课程简介,ToutiaoWeb虚拟机使用说明【附代码文档】

嘿马头条项目从到完整开发笔记总结完整教程&#xff08;附代码资料&#xff09;主要内容讲述&#xff1a;课程简介&#xff0c;ToutiaoWeb虚拟机使用说明&#xff0c;Pycharm远程开发&#xff0c;产品与开发&#xff0c;数据库1 产品介绍,2 原型图与UI图,3 技术架构,4 开发。OS…

探索大语言模型(LLM):部分数据集介绍

探索大语言模型&#xff08;LLM&#xff09;的宝库&#xff1a;精选数据集介绍 在人工智能的黄金时代&#xff0c;大语言模型&#xff08;LLM&#xff09;的发展正以惊人的速度推进。它们不仅改变了我们与机器交互的方式&#xff0c;还在持续拓展技术的边界。作为这一进程的核…

STM32第七节:GPIO输入——按键检测(包含带参宏)

目录 前言 STM32第七节&#xff1a;GPIO输入——按键检测&#xff08;包含带参宏&#xff09; 带参宏 代码替换展示 定义带参宏 GPIO输入——按键检测 硬件部分 端口输入数据寄存器&#xff08;GPIOx_IDR&#xff09; 编写程序 配置以及编写bsp_key文件 main函数编程…

5. git 删除版本标签

要删除本地的 Git 标签&#xff0c;你可以使用 git tag -d 命令&#xff0c;后面跟上你想要删除的标签名。 以下是如何操作的步骤&#xff1a; 1.打开命令行界面&#xff08;终端或命令提示符&#xff09;。 2.导航到你的 Git 仓库所在的目录。 3.使用以下命令删除标签&…