我的编程空间,编程开发者的网络收藏夹
学习永远不晚

由 JavaScript 的 with 引发的探索

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

由 JavaScript 的 with 引发的探索

一下文章来源于微信公众号前端巅峰

1. 背景

某天吃饭的时候突然想到,都说 with 会有问题,那么是什么问题,是怎样导致的呢?知其然不知其所以然,在好奇心的驱使下,从 with 出发,一路追溯到 VO、AO。那么先来复习一下 with 是干嘛的吧。

2. with

js 的 with 是为对象访问提供命名空间式的访问方式,with 创建一个对象的命名空间,在这个命名空间内你可以直接访问对象的属性,而不需要通过对象来访问:

const o = { a: 1, b: 2 };

with (o) {
    console.log(a); // 1
    b = 3; // o: { a: 1, b: 3 }
}

看起来挺方便的哈?

const o = { a: 1, b: 2 };
const p = { a: 3 };
with (o) {
    console.log(a); // 1
    with (p) {
        console.log(a); // 3
        b = 4; // o: { a: 1, b: 4 }
        c = 5; // window.c = 5
    }
}

嗯,他还有作用域链的性质。但是,如果给不存在的属性赋值,将会沿着作用域链给该变量赋值,在没有找到变量时,非严格模式下将会自动在全局创建一个变量。这就导致了数据泄露。

那么导致数据泄露的原因是什么呢?这需要了解 LHS 查询,这个待会再说。

那来看看 js 是怎么查询的:当 with 对象 o 的时候,with 声明的作用域是 o,从这里对 c 进行 LHS 查询。o 的作用域和全局作用域都没有找到 c,在非严格模式下,失败的 LHS 会自动隐式的在全局创建一个标识符 c,如果是严格模式,则会抛出 ReferenceError

2.1. with 的性能问题

使用 with:

将近 10 倍的差距。

原因是什么呢?

js 预编译阶段会进行的优化,由于 with 创建新的词法作用域,导致 o 的 a 属性和 o 分离开位于两个不同的作用域,不能快速找到标识符,引擎将不会做任何优化。

这就好比你去某家店,引擎给了你一个牛逼的老大,老板一眼就知道该怎么做;套上 with 之后,老板不知道你的老大是谁,还要花时间去找,时间就这样浪费了。

3. LHS 和 RHS

  • LHS:赋值操作的目标是谁
  • RHS:谁是赋值操作的源头

所以我们来看这段代码:

var a = 1;

在执行的时候,这段代码会被拆成两部分

var a;
a = 1;

当我们使用 a

console.log(a);

对 a 进行了 RHS 查询,以获得 a 的值,并隐式赋值给console.log 函数的形参,赋值对象的查找也是 LHS 查询。

当然,更直白的 LHS 也如上文写到的 a = 1

在变量还没有声明(在任何作用域中都无法找到该变量)情况下,这两种查询行为是不一样的。

LHS 和 RHS 查询都会在当前执行作用域中开始,沿着作用域链向上查找,直到全局作用域。

而不成功的 RHS 会抛出 ReferenceError ,不成功的 LHS 会自动隐式地创建一个全局变量(非严格模式),并作为 LHS 的查询结果(严格模式也会抛出 ReferenceError)。

那么,复习一下作用域链的查找吧。

4. 执行上下文和作用域链

在 js 中有三种代码运行环境:

  • 全局执行环境
  • 函数执行环境
  • Eval 执行环境

js 代码执行的时候,为了区分运行环境,会进入不同的执行上下文(Execution context,EC),这些执行上下文会构成一个执行上下文栈(Execution context stackECS)。

对于每个 EC 都有一个变量对象(Variable object,VO),作用域链(Scope chain)和 this 三个主要属性。

4.1. VO

变量对象(Variable object)是与 EC 相关的作用域,存储了在 EC 中定义的变量和函数声明。VO 中一般会包含以下信息:

  • 变量
  • 函数声明式
  • 函数的形参

而函数表达式和没有用 varletconst 声明的变量(全局变量,存在于全局 EC 的 VO)不存在于 VO 中。对于全局 VO,有 VO === this === global。

4.2. AO

在函数 EC 中,VO 是不能直接访问的,此时由激活对象(Activation ObjectAO)来替代 VO 的角色。AO 是在进入函数 EC 时被创建的,它通过函数的 arguments 进行初始化。这时,VO === AO。

如:

function foo(b) {
    const a = 1;
}
foo(1);
// AO:{ arguments: {...}, a: 1, b: 1 }

4.3. 作用域链

了解了 EC、AO、VO,再来看作用域链

var x = 10;
function foo() {
    var y = 20;
    
    function bar() {
        var z = 30;
       
        console.log(x + y + z);
    };
    
    bar()
};
foo();

  • 橙色箭头指向 VO/AO
  • 蓝色箭头们则是作用域链(VO/AO + All Parent VO/AOs)

理解了查找过程,很容易想到 js 中的原型链,而作用域链和原型链则可以组合成一个二维查找:先通过作用域链查找到某个对象,再查找这个对象上的属性。

const foo = {}
function bar() {
    Object.prototype.a = 'Set foo.a from prototype';
    returnfunction () {
        console.log(foo.a);
    }
}
bar()(); 
// Set foo.a from prototype

到此这篇关于由 JavaScript 的 with 引发的探索的文章就介绍到这了,更多相关 JavaScript 的 with 引发的探索内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

由 JavaScript 的 with 引发的探索

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

由JavaScript的with 引发的探索是怎样的

本篇文章给大家分享的是有关由JavaScript的with 引发的探索是怎样的,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。1. 背景某天吃饭的时候突然想到,都说 with 会
2023-06-26

探索Unity3D游戏引擎,开启游戏开发之旅(Unity3D游戏引擎在游戏开发中的应用与探索)

Unity3D是一款跨平台游戏引擎,广泛应用于各类游戏开发,包括3D、2D、移动、VR和AR游戏。其优势在于跨平台支持、强大的图形渲染、物理引擎、友好的用户界面和丰富的工具集。探索Unity3D可通过在线教程、官方文档、社区论坛、资产商店和创建游戏原型等方式,帮助开发人员深入了解引擎的工作原理并将其应用于游戏开发中。
探索Unity3D游戏引擎,开启游戏开发之旅(Unity3D游戏引擎在游戏开发中的应用与探索)
2024-04-02

JavaScript Webpack 的未来展望:探索未来的发展趋势

JavaScript Webpack 是一个功能强大的打包工具,用于将多个 JavaScript 模块捆绑到一个或多个文件中。它在前端开发中已广泛使用,并在不断更新和改进中。本文将探讨 JavaScript Webpack 的未来展望,着眼于其即将发布的功能和潜在的发展趋势。
JavaScript Webpack 的未来展望:探索未来的发展趋势
2024-02-24

javascript是由谁开发的

这篇文章主要为大家展示了“javascript是由谁开发的”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“javascript是由谁开发的”这篇文章吧。javascript是Netscape公司开
2023-06-15

探索 JavaScript BOM 的未知领域

JavaScript BOM 对象模型是用于控制和操作 Web 浏览器的内容、窗口和其他方面的 API,使用 BOM,可以访问当前加载的网页的地址、历史记录、屏幕大小、浏览器版本等信息。 JavaScript BOM、窗口对象、文档对象、历史对象、屏幕对象、导航对象 一、BOM 对象模型概述
探索 JavaScript BOM 的未知领域
2024-02-21

ES6 语法:探索 JavaScript 的新天地

ES6 语法:解锁 JavaScript 的无限可能
ES6 语法:探索 JavaScript 的新天地
2024-03-09

TypeScript赋能Vue Router:探索路由的新境界

TypeScript为Vue Router注入强类型系统和自动代码提示,增强了路由管理的可维护性和可读性。本文深入探讨TypeScript和Vue Router的集成,展示如何利用TypeScript特性创建强大的单页面应用程序。
TypeScript赋能Vue Router:探索路由的新境界
2024-02-15

throw 语句的奥秘:深入探索 JavaScript

JavaScript throw 语句:深入探索 JavaScript 异常处理的利器
throw 语句的奥秘:深入探索 JavaScript
2024-03-01

编程热搜

目录