10 个用图表解释JavaScript 闭包的面试题

news2025/1/19 7:58:16

闭包是函数式编程中的核心概念之一,是每个 JavaScript 开发人员必备的知识。在这里,我准备了 10 个关于闭包的面试挑战题,这些基本都是面试中经常被问到的。

你准备好了吗?我们现在要开始了。

每个题目都有一个代码片段,你需要说出这段代码的输出是什么。

1、范围

在说闭包之前,我们必须了解作用域的概念,它是理解闭包的基石。

此代码段的输出是什么?

var a = 10
function foo(){    console.log(a)}
foo()

这很简单,相信所有人都知道输出结果是10。

  • 默认情况下,有一个全局范围。

  • 本地作用域由函数或代码块创建。

当执行 console.log(a) 时,JavaScript 引擎将首先在函数 foo 创建的本地范围内查找 a。当 JavaScript 引擎找不到 a 时,它会尝试在其外部作用域(即全局作用域)中查找 a。然后事实证明a的值为10。

2、 局部作用域

var a = 10
function foo(){    var a = 20    console.log(a)}
a = 30
foo()

在这段代码中,变量 a 也存在于 foo 的范围内。所以当执行 console.log(a) 时,JavaScript 引擎可以直接从本地作用域获取 a 的值。

所以输出是 20 。

记住:当 JavaScript 引擎需要查询一个变量的值时,它会首先在本地范围内查找,如果没有找到该变量,它会继续在上层范围内查找。

3、词法作用域

var a = 10function foo(){    console.log(a)}
function bar() {    var a = 20    foo()}
bar()

这个问题容易出错,也是面试中经常出现的问题,你可以考虑一下。

简单地说,JavaScript 实现了一种名为词法作用域(或静态作用域)的作用域机制。它被称为词法(或静态),因为引擎仅通过查看 JavaScript 源代码来确定范围的嵌套,无论它在哪里调用。

所以输出是 10 :

4、修改词法作用域

如果我们将代码片段更改为:

var a = 10
function bar() {  var a = 20
  function foo(){    console.log(a)  }
  foo()}
bar()

输出是什么?

foo 范围成为 bar 范围的子范围:

当 JavaScript 引擎在 Foo 作用域中没有找到 a 时,它会首先从 Foo 作用域的父作用域,也就是 Bar 作用域中寻找 a,它确实找到了 a。

所以输出是 20:

好了,以上就是关于范围的一些基本挑战,相信你能顺利通过。现在我们开始进入闭包的部分。

5、 闭包

function outerFunc() {  let a = 10;
  function innerFunc() {    console.log(a);  }  return innerFunc;}
let innerFunc = outerFunc();innerFunc()

输出是什么?这段代码会抛出异常吗?

在词法范围内,innerFunc 仍然可以访问 a,即使在其词法范围之外执行。

换句话说,innerFunc 从其词法范围中记住(或关闭)变量 a。

换句话说,innerFunc 是一个闭包,因为它在变量 a 的词法范围内关闭。

因此,这段代码不会抛出异常,而是输出 10。

6、 IIFE

(function(a) {  return (function(b) {    console.log(a);  })(1);})(0);

此代码片段使用 JavaScript 立即调用函数表达式 (IIFE)。

我们可以简单地将这段代码翻译成这样:

function foo(a){  function bar(b){    console.log(a)  }  return bar(1)}
foo(0)

所以输出是 0 。

闭包的一个经典应用是隐藏变量。

比如现在要写一个计数器,基本的写法是这样的:

let i = 0function increase(){  i++  console.log(`courrent counter is ${i}`)  return i}
increase()increase()increase()

可以这样写,但是在全局范围内会多出一个变量i,这样就不好了。

这时候,我们可以使用闭包来隐藏这个变量。

let increase = (function(){  let i = 0  return function(){    i++    console.log(`courrent counter is ${i}`)    return i  }})()
increase()increase()increase()

这样,变量 i 就隐藏在局部范围内,不会污染全局环境。

7、多重声明和使用

let count = 0;
(function() {  if (count === 0) {    let count = 1;    console.log(count);  }  console.log(count);})();

在这个代码片段中,有两个 count 的声明和三个 count 的用法。这是一个难题,你应该仔细考虑。

首先,我们要知道if代码块也创建了一个局部作用域,上面的作用域大致是这样的。

  • Function Scope 没有声明自己的计数,所以我们在这个作用域中使用的计数是全局作用域的计数。

  • If Scope 声明了自己的计数,所以我们在这个作用域中使用的计数就是当前作用域的计数。

或在此图中:

所以输出是 1 , 0 :

8、调用多个闭包

function createCounter(){let i = 0return function(){    i++return i  }}let increase1 = createCounter()let increase2 = createCounter()console.log(increase1())console.log(increase1())console.log(increase2())console.log(increase2())

这里需要注意的是,increase1和increase2是通过不同的函数调用createCounter创建的,它们不共享内存,它们的i是独立的,不同的。

所以输出是 1 , 2 , 1 , 2 。

9、返回函数

function createCounter() {  let count = 0;  function increase() {     count++;  }  let message = `Count is ${count}`;
  function log() {    console.log(message);  }
  return [increase, log];}const [increase, log] = createCounter();increase(); increase(); increase(); log();

这段代码很容易理解,但是有个陷阱:message其实是一个静态字符串,它的值固定为Count为0,当我们调用increase或者log时不会改变。

所以每次调用 log 函数,输出结果总是 Count is 0 。

如果您希望 log 函数及时检查 count 的值,请将 message 移入 log :

function createCounter() {  let count = 0;  function increase() {     count++;  }-  let message = `Count is ${count}`;
  function log() {+  let message = `Count is ${count}`;    console.log(message);  }
  return [increase, log];}const [increase, log] = createCounter();increase(); increase(); increase(); log();

10、异步闭包

for (var i = 0; i < 5; i++) {  setTimeout(function () {    console.log(i);  }, 0)}

输出是什么?

上面的代码等价于:

var i = 0;setTimeout(function(){  console.log(i);},0)i = 1;setTimeout(function(){  console.log(i);},0)i = 2;setTimeout(function(){  console.log(i);},0)i = 3;setTimeout(function(){  console.log(i);},0)i = 4;setTimeout(function(){  console.log(i);},0)i = 5

而且我们知道JavaScript会先执行同步代码,然后再执行异步代码。所以每次执行console.log(i)时,i的值已经变成了5。

所以输出是 5 , 5 , 5 , 5 , 5 。

如果我们想要代码输出 0 , 1 , 2 , 3 , 4 ,需要怎么操作?

使用闭包的解决方案是:

for ( var i = 0 ; i < 5 ; ++i ) {  (function(cacheI){      setTimeout(function(){          console.log(cacheI);      },0)  })(i)}  ;

上面的代码等价于:

var i = 0;(function(cacheI){setTimeout(function(){  console.log(cacheI);},0)})(i)
i = 1;(function(cacheI){setTimeout(function(){  console.log(cacheI);},0)})(i)
i = 2;(function(cacheI){setTimeout(function(){  console.log(cacheI);},0)})(i)
i = 3;(function(cacheI){setTimeout(function(){  console.log(cacheI);},0)})(i)
i = 4;(function(cacheI){setTimeout(function(){  console.log(cacheI);},0)})(i)

我们通过 JavaScript 立即调用的函数表达式创建函数范围。i 的值是通过闭包保存的。

恭喜你,到这里,你已经学会了这些面试挑战题。

希望在开发面试中,闭包相关的问题不会再困扰你了。

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

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

相关文章

代码随想录算法训练营第五十天| LeetCode123. 买卖股票的最佳时机 III、LeetCode188. 买卖股票的最佳时机 IV

一、LeetCode123. 买卖股票的最佳时机 III 1&#xff1a;题目描述&#xff08;123. 买卖股票的最佳时机 III&#xff09; 给定一个数组&#xff0c;它的第 i 个元素是一支给定的股票在第 i 天的价格。 设计一个算法来计算你所能获取的最大利润。你最多可以完成 两笔 交易。 注意…

小学生学Arduino---------点阵(一)静态图片显示

今天来看一下&#xff0c;点阵模块这一模块可以做出非常有意思的东西。 学习目标&#xff1a; 1、了解点阵原理 2、掌握图形绘制&#xff08;心形、三角形等&#xff09; 3、掌握图形显示器的功能 4、掌握led点阵屏幕的功能 5、搭建电路 6、编写程序 一、点阵的原理 LED点阵屏…

Functional Programming in Java venkat(16) Being Lazy part3

文章目录Functional Programming in Java venkat(16): Being LazyLeveraging the Laziness of StreamsIntermediate and Terminal OperationsMethod Evaluation OrderPeeking into the LazinessFunctional Programming in Java venkat(16): Being Lazy 这里是记录学习这本书 F…

linux权限详解

文章目录1.用户转换1.将普通用户转换成root1.su -2. su2.将root转换为普通用户2.文件的权限1.文件访问者的分类拥有者和other所属组2.rwx的含义3.修改权限第一种修改方式1.拥有者修改2.所属组的修改3.other的修改4.整体修改第二种修改方式666000777使用权限的修改1.拥有者用户的…

[事务]-事务概念/特性/并发问题/传播特性

1. 事务的概念 事务&#xff08;Transaction&#xff09;指的是一个操作序列&#xff0c;该操作序列中的多个操作要么都做&#xff0c;要么都不做&#xff0c;是一个不可分割的工作单位&#xff0c;是数据库环境中的逻辑工作单位&#xff0c;由DBMS中的事务管理子系统负责…

爬虫工作流程、请求与响应原理、requests库讲解

爬虫工作流程、请求与响应原理、requests库讲解 爬虫分类主要分为两大板块 web爬虫&#xff08;浏览器爬虫&#xff09; APP爬虫&#xff08;手机端爬虫&#xff09; 在这两大板块中又可以把爬虫归类为聚焦爬虫和通用爬虫 聚焦爬虫&#xff1a;针对某一个接口&#xff08;ur…

对话框被遮罩层挡住

element-ui 解决方法一&#xff1a; 在el-dialog中写去掉遮罩层 :modal"false" 解决方法二&#xff1a; 在el-dialog中写&#xff08;遮罩层是否插入至 body 元素上&#xff0c;若为 false&#xff0c;则遮罩层会插入至 Dialog 的父元素上&#xff09; :modal-ap…

[附源码]SSM计算机毕业设计医院挂号系统JAVA

项目运行 环境配置&#xff1a; Jdk1.8 Tomcat7.0 Mysql HBuilderX&#xff08;Webstorm也行&#xff09; Eclispe&#xff08;IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持&#xff09;。 项目技术&#xff1a; SSM mybatis Maven Vue 等等组成&#xff0c;B/S模式 M…

专业硕士招生占比将达到三分之二,那么跟学术硕士有哪些区别?

近年来的研究生招生考试中&#xff0c;专业硕士的招生培养规模正稳步增长。据统计&#xff0c;2009年专业学位硕士招生人数在硕士招生总人数中的占比仅为15.9%&#xff0c;其后在2017年首次超过学硕招生人数&#xff0c;到2020年专硕招生人数占比已超60%。国务院学位委员会、教…

SpringBoot - 集成Actuator(应用信息显示、修改系统日志、增加账号密码登录)

文章目录Actuator概述官网入口支持的埋点信息查询、修改使用访问actuator埋点信息添加账号密码登录验证动态修改日志级别Actuator 概述 官网入口 官网&#xff1a; https://docs.spring.io/spring-boot/docs/2.7.6/reference/html/actuator.html#actuator.endpoints 支持的埋…

csdn中书写数学公式简单介绍

参考&#xff1a;https://www.zybuluo.com/codeep/note/163962#3%E5%9C%A8%E5%AD%97%E7%AC%A6%E9%97%B4%E5%8A%A0%E5%85%A5%E7%A9%BA%E6%A0%BC 常识、常用 一行公式使用$$开始和结尾&#xff0c;常用符号表示 符号功能$$多行公式的开始和结尾&#xff0c;一个$表示单行公式开…

Cisco ASA基础——安全算法与基本配置

作者简介&#xff1a;一名在校云计算网络运维学生、每天分享网络运维的学习经验、和学习笔记。 座右铭&#xff1a;低头赶路&#xff0c;敬事如仪 个人主页&#xff1a;网络豆的主页​​​​​​ 目录 前言 本章重点 一.Cisco防火墙简介 1.什么是防火墙 2.防火墙的作用…

PHP表单处理的案例分析

目录 知识补充 实现过程 前端代码 后端代码 简单分析 知识补充 表单简介&#xff08;来自Mr._Dang&#xff09; action&#xff1a;提交的地址 method&#xff1a;提交的方式 get&#xff1a; 参数是在url中的&#xff0c;不安全&#xff0c;传输量比较少&#xff…

[附源码]Python计算机毕业设计Django的在线作业批改系统

项目运行 环境配置&#xff1a; Pychram社区版 python3.7.7 Mysql5.7 HBuilderXlist pipNavicat11Djangonodejs。 项目技术&#xff1a; django python Vue 等等组成&#xff0c;B/S模式 pychram管理等等。 环境需要 1.运行环境&#xff1a;最好是python3.7.7&#xff0c;…

[附源码]Python计算机毕业设计SSM力高灯饰线上交易平台(程序+LW)

项目运行 环境配置&#xff1a; Jdk1.8 Tomcat7.0 Mysql HBuilderX&#xff08;Webstorm也行&#xff09; Eclispe&#xff08;IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持&#xff09;。 项目技术&#xff1a; SSM mybatis Maven Vue 等等组成&#xff0c;B/S模式 M…

一言不合就重构

hello&#xff0c;大家好呀&#xff0c;我是小楼。 前段时间不是在忙么&#xff0c;忙的内容之一就是花了点时间重构了一个服务的健康检查组件&#xff0c;目前已经慢慢在灰度线上&#xff0c;本文就来分享下这次重构之旅&#xff0c;也算作个总结吧。 背景 服务健康检查简介…

短视频创作,变现的建议、变现方式和举例,建议收藏反复阅读-上

先说今天的纲要&#xff0c;有兴趣可以继续看下去&#xff0c;今天主要针对短视频变现这件事的讨论&#xff0c;有三个建议&#xff0c;①变现标准低、②变现天花板高、③可主动变现。 我们在选择变现形式的时候&#xff0c;尽可能满足这三个条件或其中两个。 中间我们再讨论下…

【LeetCode】895.最大频率栈

题目描述 设计一个类似堆栈的数据结构&#xff0c;将元素推入堆栈&#xff0c;并从堆栈中弹出出现频率最高的元素。 实现 FreqStack 类: FreqStack() 构造一个空的堆栈。void push(int val) 将一个整数 val 压入栈顶。int pop() 删除并返回堆栈中出现频率最高的元素。 如果出现…

【问题思考总结】NAT的公有地址怎么转换为私有地址?【MAC地址和IP地址的转换】

问题起源 在做一道题的时候&#xff0c;涉及到了由内网到外网再到内网时的IP地址转换。在外网的时候&#xff0c;答案说的是不能够用私有IP地址作为源IP地址&#xff0c;然后疑问产生了&#xff1a;如果不能用私有IP地址作为目的地址&#xff0c;他又怎么能够找到那个主机呢&a…

[附源码]Python计算机毕业设计SSM乐多多宠物店网站(程序+LW)

项目运行 环境配置&#xff1a; Jdk1.8 Tomcat7.0 Mysql HBuilderX&#xff08;Webstorm也行&#xff09; Eclispe&#xff08;IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持&#xff09;。 项目技术&#xff1a; SSM mybatis Maven Vue 等等组成&#xff0c;B/S模式 M…