热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

重学前端学习笔记(十八)JavaScript的闭包和执行上下文

笔记说明重学前端是程劭非(winter)【前手机淘宝前端负责人】在极客时间开的一个专栏,每天10分钟,重构你的前端知识体系,笔者主要整理学习过程的一些要点笔记以及感悟,完整的可以加

笔记说明

重学前端是程劭非(winter)【前手机淘宝前端负责人】在极客时间开的一个专栏,
每天10分钟,重构你的前端知识体系,笔者主要整理学习过程的一些要点笔记以及感悟,完整的可以加入winter的专栏学习【原文有winter的语音】,如有侵权请联系我,邮箱:kaimo313@foxmail.com。

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

《重学前端学习笔记(十八)--Javascript的闭包和执行上下文》

二、闭包(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 的特性会导致声明的变量和被赋值的变量是两个 bJavascript 中有特例,那就是使用 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 查询

LHS
Left-hand Side)引用和
RHS
Right-hand Side)引用。通常是指等号(赋值运算)的左右边的引用。

console.log(gg)

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

gg = 666;

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

综上

1、当传递 k2with 时,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

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

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 中执行,所以表现出了不同的行为。

个人总结

总的来说,这一篇一脸懵逼(_(:3」∠)_),要去看看《你不知道的Javascript》才行了。。。


推荐阅读
  • 本文探讨了如何使用Scrapy框架构建高效的数据采集系统,以及如何通过异步处理技术提升数据存储的效率。同时,文章还介绍了针对不同网站采用的不同采集策略。 ... [详细]
  • 本文详细介绍如何在SSM(Spring + Spring MVC + MyBatis)框架中实现分页功能。包括分页的基本概念、数据准备、前端分页栏的设计与实现、后端分页逻辑的编写以及最终的测试步骤。 ... [详细]
  • 本文旨在探讨Swift中的Closure与Objective-C中的Block之间的区别与联系,通过定义、使用方式以及外部变量捕获等方面的比较,帮助开发者更好地理解这两种机制的特点及应用场景。 ... [详细]
  • 解析Java虚拟机HotSpot中的GC算法实现
    本文探讨了Java虚拟机(JVM)中HotSpot实现的垃圾回收(GC)算法,重点介绍了根节点枚举、安全点及安全区域的概念和技术细节,以及这些机制如何影响GC的效率和准确性。 ... [详细]
  • Java虚拟机及其发展历程
    Java虚拟机(JVM)是每个Java开发者日常工作中不可或缺的一部分,但其背后的运作机制却往往显得神秘莫测。本文将探讨Java及其虚拟机的发展历程,帮助读者深入了解这一关键技术。 ... [详细]
  • 深入解析Unity3D游戏开发中的音频播放技术
    在游戏开发中,音频播放是提升玩家沉浸感的关键因素之一。本文将探讨如何在Unity3D中高效地管理和播放不同类型的游戏音频,包括背景音乐和效果音效,并介绍实现这些功能的具体步骤。 ... [详细]
  • 本文回顾了作者在求职阿里和腾讯实习生过程中,从最初的迷茫到最后成功获得Offer的心路历程。文中不仅分享了个人的面试经历,还提供了宝贵的面试准备建议和技巧。 ... [详细]
  • egg实现登录鉴权(七):权限管理
    权限管理包含三部分:访问页面的权限,操作功能的权限和获取数据权限。页面权限:登录用户所属角色的可访问页面的权限功能权限:登录用户所属角色的可访问页面的操作权限数据权限:登录用户所属 ... [详细]
  • 本文介绍了Tomcat的基本操作,包括启动、关闭及首次访问的方法,并详细讲解了如何在IDEA中创建Web项目,配置Servlet及其映射,以及如何将项目部署到Tomcat。 ... [详细]
  • SSE图像算法优化系列三:超高速导向滤波实现过程纪要(欢迎挑战)
    自从何凯明提出导向滤波后,因为其算法的简单性和有效性,该算法得到了广泛的应用,以至于新版的matlab都将其作为标准自带的函数之一了&#x ... [详细]
  • 本文详细探讨了Java中HashMap类的hash()方法的工作原理及其重要性,特别是在JDK 7版本中的实现。 ... [详细]
  • Spring Security基础配置详解
    本文详细介绍了Spring Security的基础配置方法,包括如何搭建Maven多模块工程以及具体的安全配置步骤,帮助开发者更好地理解和应用这一强大的安全框架。 ... [详细]
  • 在开发过程中,有时需要提供用户创建数据库的功能。本文介绍了如何利用 .NET 和 ADOX 在应用程序中实现创建 Access 数据库,并详细说明了创建数据库及表的具体步骤。 ... [详细]
  • 深入理解线程池及其基本实现
    本文探讨了线程池的概念、优势及其在Java中的应用。通过实例分析不同类型的线程池,并指导如何构建一个简易的线程池。 ... [详细]
  • Asynchronous JavaScript and XML (AJAX) 的流行很大程度上得益于 Google 在其产品如 Google Suggest 和 Google Maps 中的应用。本文将深入探讨 AJAX 在 .NET 环境下的工作原理及其实现方法。 ... [详细]
author-avatar
dancejust25679
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有