深入理解JS的执行上下文、词法作用域和闭包(下)

在这里插入图片描述

🤍 前端开发工程师、技术日更博主、已过CET6
🍨 阿珊和她的猫_CSDN博客专家、23年度博客之星前端领域TOP1
🕠 牛客高级专题作者、打造专栏《前端面试必备》 、《2024面试高频手撕题》
🍚 蓝桥云课签约作者、上架课程《Vue.js 和 Egg.js 开发企业级健康管理项目》、《带你从入门到实战全面掌握 uni-app》

文章目录

  • 四、总结 🎯
    • 执行上下文、词法作用域和闭包的关系
    • 理解这些概念对编程的重要性 💪
    • 实际应用中的注意事项和最佳实践 👀
  • 参考资料

四、总结 🎯

执行上下文、词法作用域和闭包的关系

在JavaScript中,执行上下文、词法作用域和闭包之间存在一些复杂的关系。下面详细解释这些关系:

执行上下文:

执行上下文是JavaScript中变量、函数和对象等执行时的上下文环境。JavaScript中,执行上下文主要包括全局执行上下文(Global Execution Context)和函数执行上下文(Functional Execution Context)。

全局执行上下文包含全局作用域中的变量、函数和对象等。函数执行上下文则包含函数词法作用域中的变量、函数和对象等。

当执行一段代码时,首先会创建全局执行上下文,将全局变量、函数和对象等放入其中。然后,按照代码中的函数调用顺序,依次创建函数执行上下文,并将函数词法作用域中的变量、函数和对象等放入其中。

词法作用域:

词法作用域是由代码文本中变量声明的位置决定的。在JavaScript中,词法作用域主要包括函数词法作用域和块级作用域。

函数词法作用域是指函数可以访问其词法作用域中的变量,即使这个函数在其词法作用域之外执行。块级作用域是指使用letconst声明的变量具有块级作用域,只能在声明它们的代码块(如for循环、if语句等)内部访问。

词法作用域主要影响变量的查找和函数的调用。当需要在某个作用域中查找变量或调用函数时,JavaScript会按照词法作用域的规则,从当前作用域开始,向上级作用域逐层查找,直到找到为止。

闭包:

闭包(Closure)是JavaScript中一种重要的概念,它指的是一个函数可以访问其词法作用域中的变量,即使这个函数在其词法作用域之外执行。

闭包的主要特点包括函数可以记住并访问其词法作用域,即使这个作用域在函数执行完毕后被销毁。当一个函数在其词法作用域之外执行时,即使这个作用域在函数执行完毕后被销毁,该函数仍然可以访问其词法作用域中的变量。

总之,执行上下文、词法作用域和闭包之间的关系主要体现在以下几个方面:

  • 执行上下文是JavaScript中变量、函数和对象等执行时的上下文环境,词法作用域主要影响变量的查找和函数的调用。
  • 词法作用域是由代码文本中变量声明的位置决定的,闭包指的是一个函数可以访问其词法作用域中的变量,即使这个函数在其词法作用域之外执行。
  • 闭包可以让函数记住并访问其词法作用域,即使这个作用域在函数执行完毕后被销毁。

实际项目中,可以根据具体需求和场景,灵活地使用执行上下文、词法作用域和闭包等概念。

理解这些概念对编程的重要性 💪

理解执行上下文、词法作用域和闭包对编程的重要性主要体现在以下几个方面:

提高代码的可读性和可维护性:理解这些概念有助于更好地理解JavaScript中的变量作用域和函数调用等问题,从而提高代码的可读性和可维护性。

例如,理解词法作用域可以让我们更好地理解以下代码:

function foo() {let a = 1;function bar() {console.log(a); // 输出 1}bar();
}foo();

在这个例子中,bar函数在其词法作用域foo之外执行,但仍然可以访问foo作用域中的变量a。理解词法作用域有助于我们更好地理解这个例子。

避免变量污染和冲突:理解执行上下文和词法作用域可以帮助我们更好地管理变量的作用域,避免全局作用域和函数作用域中的变量冲突。

例如,以下代码会导致全局作用域和函数作用域中的变量冲突:

let a = 1;function foo() {let a = 2;console.log(a); // 输出 2
}foo();
console.log(a); // 输出 2

在这个例子中,全局作用域和函数作用域中的变量a冲突,导致输出结果不符合预期。理解执行上下文和词法作用域有助于我们避免这种错误。

实现模块化和封装:闭包可以让函数记住并访问其词法作用域,即使这个作用域在函数执行完毕后被销毁。这使得我们可以将一组相关的函数和变量封装在一个函数内部,然后返回这个函数的引用,这样就可以在其他地方调用这些函数并访问它们需要的变量。

例如,使用闭包实现一个简单的模块:

function createCart() {let products = [];function addProduct(product) {products.push(product);}function getProducts() {return products;}return {addProduct,getProducts};
}const cart = createCart();
cart.addProduct({ id: 1, name: 'Product 1' });
console.log(cart.getProducts()); // 输出:[{ id: 1, name: 'Product 1' }]

在这个例子中,createCart函数返回一个闭包,包含addProductgetProducts函数以及它们需要的变量products。这样可以实现模块化,将一组相关的函数和变量封装在一个函数内部。

总之,理解执行上下文、词法作用域和闭包对编程的重要性主要体现在提高代码的可读性和可维护性、避免变量污染和冲突以及实现模块化和封装等方面。实际项目中,可以根据具体需求和场景,灵活地使用这些概念。

实际应用中的注意事项和最佳实践 👀

在实际应用中,关于执行上下文、词法作用域和闭包,有一些注意事项和最佳实践:

使用letconst声明变量:在声明变量时,建议使用letconst而不是varletconst声明的变量具有块级作用域,可以在声明它们的代码块(如for循环、if语句等)内部访问,而var声明的变量具有函数作用域,可以在整个函数内部访问。

例如,以下代码使用let声明变量:

function foo() {let a = 1;function bar() {console.log(a); // 输出 1}bar();
}foo();

而以下代码使用var声明变量:

function foo() {var a = 1;function bar() {console.log(a); // 输出 1}bar();
}foo();

使用函数执行上下文:在实际项目中,可以使用函数执行上下文来封装一组相关的函数和变量,这样可以实现模块化和封装。

例如,以下代码使用函数执行上下文:

function createCart() {let products = [];function addProduct(product) {products.push(product);}function getProducts() {return products;}return {addProduct,getProducts};
}const cart = createCart();
cart.addProduct({ id: 1, name: 'Product 1' });
console.log(cart.getProducts()); // 输出:[{ id: 1, name: 'Product 1' }]

避免全局作用域污染:在实际项目中,避免全局作用域污染非常重要。全局作用域中的变量和函数可能会影响到其他代码,导致难以维护和调试。可以使用闭包来避免全局作用域污染。

例如,以下代码使用闭包避免全局作用域污染:

function createCounter() {let count = 0;return function() {count += 1;console.log(count); // 输出 1 2 3};
}const counter = createCounter();
counter();
counter();
counter();

总之,在实际应用中,关于执行上下文、词法作用域和闭包,需要注意使用letconst声明变量,使用函数执行上下文封装相关函数和变量,以及避免全局作用域污染。这些注意事项和最佳实践有助于提高代码的可读性和可维护性,降低代码的耦合度,使代码更加清晰和易于维护。

参考资料

[1]《JavaScript 高级程序设计》
[2] MDN Web Docs: JavaScript

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

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

相关文章

mvn版本导致的Failed to execute goal on project问题与解决

目录 一,报错情况与原因二,maven的下载与安装1,卸载maven2,安装mvn3,指定mvn的默认版本: 一,报错情况与原因 使用命令mvn package时会报如下错误: Failed to execute goal on proj…

prometheus+grafana监控nginx的简单实现

1.编译安装NGINX 加入编译安装nginx-module-vts模块,目的是为了获取更多的监控数据(虚拟主机,upstream等) nginx下载 http://nginx.org/download/nginx-1.20.2.tar.gz nginx-module-vts下载 https://github.com/vozlt/nginx-module-vts/archive/refs/tags/v0.2…

9.5K Star,又一款超棒开源轻量自动化运维平台

Hi,骚年,我是大 G,公众号「GitHub指北」会推荐 GitHub 上有趣有用的项目,一分钟 get 一个优秀的开源项目,挖掘开源的价值,欢迎关注。 一个好的运维平台就变得非常重要了,可以节省大量的人力和物…

冯诺依曼体系结构 与 操作系统

一、冯诺依曼体系结构 深入理解冯诺依曼体系结构 计算机的出现就是为了解决实际问题, 所以把问题交给计算机,计算机经过处理,得到一个结果反馈给我们,所以这中间就必然涉及到了输入设备,中央处理器(包括运算器和控制器)和输出设备…

【读后感】《枪炮、病菌与钢铁》人类社会的命运

初看这个书名其实感到困惑,风马牛不相及的三个名词怎么就凑到一起了,这书是讲什么的。 先说结论,讲的是人类历史,具体是1.3万年前开始至今的历史以及现代世界格局的形成,1.3万年前从这个时间节点以后,不论…

勇宝趣学JavaScript ES6第二章(解构赋值)

在ES6中,我们经常使用到解构赋值这个知识点,今天我们就来好好讲讲这一小部分知识点。咱们争取这回一次性讲明白。 今天是元宵节,祝大家节日快乐,只有我自己还在无聊的码字。 给我点个赞吧,嘿嘿!&#xff01…

MySQL数据库进阶第四篇(视图/存储过程/触发器)

文章目录 一、视图简单介绍与基础语法二、视图的检查选项三、视图的更新四、视图的作用五、存储过程的概念与特点六、存储过程的 创建,调用,查看,删除七、存储过程 — 系统变量八、存储过程 — 用户定义变量九、存储过程 — 局部变量十、存储…

acwing算法学习笔记 ------ 双链表

1、定义 这里可以做一个投机取巧,我们不再像单链表去用head去存头和尾,直接让r[0] 1,l[1] 0; idx 2.进行初始化, 解释一下l[N] 和 r[N] l[N]:是表示指向左面下一个节点下标, r[N]:表示指向下一个节点的下标。大家不用担心i…

[VNCTF2024]-PWN:shellcode_master解析(orw,用mmap代替read读文件)

查看保护 查看ida 在sandbox函数中,函数先使用了seccomp_init初始化,允许了所有系统调用,再用seccomp_rule_add来禁用掉了部分系统调用,其中包括execve和read seccomp_init函数可以进行系统调用全禁用和全允许初始化 seccomp_ru…

《高质量的C/C++编程规范》学习

目录 一、编程规范基础知识 1、头文件 2、程序的板式风格 3、命名规则 二、表达式和基本语句 1、运算符的优先级 2、复合表达式 3、if语句 4、循环语句的效率 5、for循环语句 6、switch语句 三、常量 1、#define和const比较 2、常量定义规则 四、函数设计 1、参…

python_pyecharts绘制漏斗图

python-pyecharts绘制漏斗图 from pyecharts.charts import Funnel from pyecharts import options as opts# 数据 data [("访问", 100), ("咨询", 80), ("订单", 60), ("点击", 40), ("展现", 20)]# 创建漏斗图 funnel …

uvloop,一个强大的 Python 异步IO编程库!

前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站零基础入门的AI学习网站~。 目录 ​编辑 前言 什么是uvloop库? 安装uvloop库 使用uvloop库 uvloop库的功能特性 1. 更…

【信息提取】FindSomething 浏览器插件

下载地址 FindSomething 浏览器插件 概述 在网页的源代码或js中找到一些有趣的东西 FindSomething 用于快速在网页的html源码或js代码中提取一些有趣的信息,包括可能请求的资源、接口的url,可能请求的ip和域名,泄漏的证件号、手机号、邮箱…

程序员可以做什么副业呢?

如果你经常玩知乎、看公众号(软件、工具、互联网这几类的)你就会发现,好多资源连接都变成了夸克网盘、迅雷网盘的资源链接。 例如:天涯神贴,基本上全是夸克、UC、迅雷网盘的资源链接。 有资源的前提下,迅雷…

2024年云南事业单位报名流程!明天就开始报名啦,千万不要错过哦

注意啦!注意啦!2024年云南事业单位报名即将开始! ▶️公告已发布,2月26日上午9:00开始报名‼️ 相关时间节点 **报名时间:**2024年2月26日9:00至3月1日18:00 **资格初审时间:**2024年2月26日…

【Python】Windows本地映射远程Linux服务器上的端口(解决jupyter notebook无法启动问题)

创作日志: 学习深度学习不想在本地破电脑上再安装各种软件,我就用实验室的服务器配置环境,启动jupyter notebook时脑子又瓦特了,在自己Windows电脑上打开服务器提供的网址,那肯定打不开啊,以前在其它电脑上…

【服务发现--service】

1、service的定义 "Service"简写"svc”。Pod不能直接提供给外网访问,而是应该使用service。Service就是把Pod暴露出来提供服务,Service才是真正的“服务”,它的中文名就叫“服务”。可以说Service是一个应用服务的抽象&#…

校园微社区微信小程序源码/二手交易/兼职交友微信小程序源码

云开发校园微社区微信小程序开源源码,这是一款云开发校园微社区-二手交易_兼职_交友_项目微信小程序开源源码,可以给你提供快捷方便的校园生活,有很多有趣实用的板块和功能,如:闲置交易、表白交友、疑问互答、任务兼职…

C++ //练习 8.9 使用你为8.1.2节(第281页)第一个练习所编写的函数打印一个istringstream对象的内容。

C Primer(第5版) 练习 8.9 练习 8.9 使用你为8.1.2节(第281页)第一个练习所编写的函数打印一个istringstream对象的内容。 环境:Linux Ubuntu(云服务器) 工具:vim 代码块 /*****…

NATS学习笔记(一)

NATS是什么? NATS是一个开源的、轻量级、高性能的消息传递系统,它基于发布/订阅模式,由Apcera公司开发和维护。 NATS的功能 发布/订阅:NATS的核心是一个发布/订阅消息传递系统,允许消息生产者发布消息到特定的主题…