重学前端 18 # JavaScript的闭包和执行上下文

简介: 重学前端 18 # JavaScript的闭包和执行上下文

一、函数执行过程相关知识


da91bea0e44baf4b93211ac967744c2f.png


二、闭包(closure)


闭包其实只是一个绑定了执行环境的函数,闭包与普通函数的区别是,它携带了执行的环境,就像人在外星中需要自带吸氧的装备一样,这个函数也带有在程序中生存的环境。


2.1、古典的闭包


  • 环境部分    
  • 环境
  • 标识符列表
  • 表达式部分



2.2、JavaScript 中闭包


  • 环境部分    
  • 环境:函数的词法环境(执行上下文的一部分)
  • 标识符列表:函数中用到的未申明的变量
  • 表达式部分:函数体



三、执行上下文(执行的基础设施)


定义:JavaScript 标准把一段代码(包括函数),执行所需的所有信息定义为执行上下文。


3.1、在 ES3


  • scope:作用域,也常常被叫做作用域链
  • variable object:变量对象,用于存储变量的对象
  • this value:this 值


3.2、在 ES5

  • lexical environment:词法环境,当获取变量时使用
  • variable environment:变量环境,当声明变量时使用
  • this value:this 值


3.3、在 ES2018 中

   lexical environment:词法环境,当获取变量或者 this 值时使用

   variable environment:变量环境,当声明变量时使用

   code evaluation state:用于恢复代码执行位置

   Function:执行的任务是函数时使用,表示正在被执行的函数

   ScriptOrModule:执行的任务是脚本或者模块时使用,表示正在被执行的代码

   Realm:使用的基础库和内置对象实例

   Generator:仅生成器上下文有这个属性,表示当前生成器



3.4、函数执行过程所需信息

var b = {}
let c = 1
this.a = 2;



正确执行上面代码需知道的信息:

  • 1、varb 声明到哪里
  • 2、b 表示哪个变量
  • 3、b 的原型是哪个对象
  • 4、letc 声明到哪里
  • 5、this 指向哪个对象


而这些信息就是执行上下文给出来的,下面用 var 声明与赋值,letrealm分析执行上下文提供的信息。


3.5、var 申明与赋值

1、立即执行的函数表达式(IIFE),通过创建一个函数,并且立即执行,来构造一个新的域,从而控制 var 的范围。

var b = 1


2、加括号让函数变成函数表达式

(function(){
    var a;
    //code
}());
(function(){
    var a;
    //code
})();


注意:括号有个缺点,那就是如果上一行代码不写分号,括号会被解释为上一行代码最末的函数调用。

一些不加分号的代码风格规范,会要求在括号前面加上分号。

;(function(){
    var a;
    //code
}())
;(function(){
    var a;
    //code
})()


winter 推荐用 void 关键字,语义上 void 运算表示忽略后面表达式的值,变成 undefined

void function(){
    var a;
    //code
}();


特别注意:var 的特性会导致声明的变量和被赋值的变量是两个 b,JavaScript 中有特例,那就是使用 with 的时候,如代码块二,我们先讲一下代码一

   with 语句的原本用意是为逐级的对象访问提供命名空间式的速写方式. 也就是在指定的代码区域, 直接通过节点名称调用对象。


// 代码块一
var obj = {
    a: 1,
    b: 2,
    c: 3
};
// 比如要改对应的值,一般的写法是重复写了3次obj
obj.a = 5;
obj.b = 6;
obj.c = 7;
console.log(obj) // {a: 5, b: 6, c: 7}
// 用 with 快捷方式
with (obj) {
    a = 5;
    b = 6;
    c = 7;
}
console.log(obj) // {a: 5, b: 6, c: 7}
// 接下来看一下 with 导致的数据泄露
function kaimo(obj) {
    with (obj) {
        a = 1;
    }
}
var k1 = {
    a: 2
};
var k2 = {
    b: 3
}
kaimo(k1);
console.log(k1.a); // 1
kaimo(k2);
console.log(k2.a); // undefined
console.log(a); // 1 (a被泄漏到全局作用域上)


上述代码分析:


   1、创建了 k1 、k2 两个对象。其中一个有 a 属性,另外一个没有。

   2、kaimo(obj) 函数接受一个 obj 的形参,该参数是一个对象引用,并执行了 with(obj) {...}。

   3、在 with 块内部,将 2 赋值给了 a。

   4、将 k1 传递进去,a = 2 赋值操作找到了 k1.a 并将 2 赋值给它。

   5、当 k2 传递进去,k2 并没有 a 的属性,因此不会创建这个属性,k2.a 保持 undefined。



问题:为什么对 k2 的操作会导致数据的泄漏呢?

首先稍微讲一下:JavaScript中的 LHSRHS 查询


LHSLeft-hand Side)引用和 RHSRight-hand Side)引用。通常是指等号(赋值运算)的左右边的引用。

console.log(gg)


比如上面这个打印,先查找并取得 gg 的值,然后将它打印出来 gg 的引用是一个 RHS 引用,没有赋予操作

gg = 666;



上面是对 gg 的引用是一个 LHS 引用,为赋值操作找到目标


综上:


1、当传递 k2 给 with 时,with 所声明的作用域是 k2, 从这个作用域开始对 a 进行 LHS 查询。


2、k2 的作用域、kaimo(…) 的作用域和全局作用域中都没有找到标识符 a,因此在非严格模式下,会自动在全局作用域创建一个全局变量),在严格模式下,会抛出 ReferenceError 异常。

// 代码块二
var b;
void function(){
    var env = {b:1};
    b = 2;
    console.log("In function b:", b);
    with(env) {
        var b = 3;
        console.log("In with b:", b);
    }
}();
console.log("Global b:", b);
// 输出结果如下:
// In function b: 2
// In with b: 3
// Global b: undefined


3.6、let

letES6 开始引入的新的变量声明模式。

winter 简单统计了下,以下语句会产生 let 使用的作用域:

for、 if、 switch、 try/catch/finally。



3.7、Realm

在最新的标准(9.0)中,JavaScript 引入了一个新概念 Realm。有道词典上的意思是:“领域,范围;王国”。Realm 中包含一组完整的内置对象,而且是复制关系。

var iframe = document.createElement('iframe')
document.documentElement.appendChild(iframe)
iframe.src="javascript:var b = {};"
var b1 = iframe.contentWindow.b;
var b2 = {};
console.log(b1, b2);
// {} {}
console.log(typeof b1, typeof b2);
// 谷歌输出: object object   火狐输出:undefined object
console.log(b1 instanceof Object, b2 instanceof Object);
//false true


上面代码可以看到,在浏览器环境中获取来自两个 Realm 的对象,由于 b1、 b2 由同样的代码 {} 在不同的 Realm 中执行,所以表现出了不同的行为。


目录
打赏
0
0
0
0
21
分享
相关文章
【01】对APP进行语言包功能开发-APP自动识别地区ip后分配对应的语言功能复杂吗?-成熟app项目语言包功能定制开发-前端以uniapp-基于vue.js后端以laravel基于php为例项目实战-优雅草卓伊凡
【01】对APP进行语言包功能开发-APP自动识别地区ip后分配对应的语言功能复杂吗?-成熟app项目语言包功能定制开发-前端以uniapp-基于vue.js后端以laravel基于php为例项目实战-优雅草卓伊凡
148 72
【01】对APP进行语言包功能开发-APP自动识别地区ip后分配对应的语言功能复杂吗?-成熟app项目语言包功能定制开发-前端以uniapp-基于vue.js后端以laravel基于php为例项目实战-优雅草卓伊凡
58K star!这个让网页动起来的JS库,前端工程师直呼真香!
Anime.js 是一款轻量级但功能强大的JavaScript动画引擎,它能够以最简单的方式为网页元素添加令人惊艳的动效。这个项目在GitHub上已经获得58,000+星标,被广泛应用于电商页面、数据可视化、游戏开发等场景。
58 8
前端开发必备!Node.js 18.x LTS保姆级安装教程(附国内镜像源配置)
本文详细介绍了Node.js的安装与配置流程,涵盖环境准备、版本选择(推荐LTS版v18.x)、安装步骤(路径设置、组件选择)、环境验证(命令测试、镜像加速)及常见问题解决方法。同时推荐开发工具链,如VS Code、Yarn等,并提供常用全局包安装指南,帮助开发者快速搭建高效稳定的JavaScript开发环境。内容基于官方正版软件,确保合规性与安全性。
687 24
JavaScript闭包深入剖析:性能剖析与优化技巧
JavaScript 闭包是强大而灵活的特性,广泛应用于数据封装、函数柯里化和事件处理等场景。闭包通过保存外部作用域的变量,实现了私有变量和方法的创建,提升了代码的安全性和可维护性。然而,闭包也可能带来性能问题,如内存泄漏和执行效率下降。为优化闭包性能,建议采取以下策略:及时解除对不再使用的闭包变量的引用,减少闭包的创建次数,使用 WeakMap 管理弱引用,以及优化闭包结构以减少作用域链查找的开销。在实际开发中,无论是 Web 前端还是 Node.js 后端,这些优化措施都能显著提升程序的性能和稳定性。
142 70
当面试官再问我JS闭包时,我能答出来的都在这里了。
闭包(Closure)是前端面试中的高频考点,广泛应用于函数式编程中。它不仅指函数内部定义的函数,还涉及内存管理、作用域链和垃圾回收机制。闭包可以让函数访问其外部作用域的变量,但也可能引发内存泄漏等问题。通过合理使用闭包,可以实现模块化、高阶函数和回调函数等应用场景。然而,滥用闭包可能导致代码复杂度增加、调试困难以及潜在的性能问题。为了避免这些问题,开发时应谨慎处理闭包,避免不必要的嵌套,并及时清理不再使用的变量和监听器。
147 16
当面试官再问我JS闭包时,我能答出来的都在这里了。
|
2月前
|
【2025优雅草开源计划进行中01】-针对web前端开发初学者使用-优雅草科技官网-纯静态页面html+css+JavaScript可直接下载使用-开源-首页为优雅草吴银满工程师原创-优雅草卓伊凡发布
【2025优雅草开源计划进行中01】-针对web前端开发初学者使用-优雅草科技官网-纯静态页面html+css+JavaScript可直接下载使用-开源-首页为优雅草吴银满工程师原创-优雅草卓伊凡发布
78 1
【2025优雅草开源计划进行中01】-针对web前端开发初学者使用-优雅草科技官网-纯静态页面html+css+JavaScript可直接下载使用-开源-首页为优雅草吴银满工程师原创-优雅草卓伊凡发布
JavaScript中闭包详解+举例,闭包的各种实践场景:高级技巧与实用指南
闭包是JavaScript中不可或缺的部分,它不仅可以增强代码的可维护性,还能在模块化、回调处理等场景中发挥巨大作用。然而,闭包的强大也意味着需要谨慎使用,避免潜在的性能问题和内存泄漏。通过对闭包原理的深入理解以及在实际项目中的灵活应用,你将能够更加高效地编写出简洁且功能强大的代码。 只有锻炼思维才能可持续地解决问题,只有思维才是真正值得学习和分享的核心要素。如果这篇博客能给您带来一点帮助,麻烦您点个赞支持一下,还可以收藏起来以备不时之需,有疑问和错误欢迎在评论区指出~
前端学习笔记202305学习笔记第三十三天-js-执行上下文
前端学习笔记202305学习笔记第三十三天-js-执行上下文
42 0
AI助理

你好,我是AI助理

可以解答问题、推荐解决方案等