由 JavaScript 的 with 引发的探索

目录
  • 1. 背景
  • 2. with
    • 2.1. with 的性能问题
  • 3. LHS 和 RHS
  • 4. 执行上下文和作用域链
    • 4.1. VO
    • 4.2. AO
    • 4.3. 作用域链

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

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:

function f () {
    const o = { a: 1 }
    console.time();

    with(o) {  
        for (let i = 0; i < 100000; i++) {
            a = 2;
        }
    }

    console.timeEnd();
}

正常情况:

function f () {
    const o = { a: 1 }
    console.time();

    for (let i = 0; i < 100000; i++) {
        o.a = 2;
    }

    console.timeEnd();
}

将近 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 引发的探索内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • javascript中的with语句学习笔记及用法

    在JS中,with 语句的作用是将代码的作用域设置到一个特定的对象中.其语法如下:with (expression) statement; 定义 with 语句的目的主要是为了简化多次编写同一个对象的工作,如下面的代码: var qs = location.search.substring(1); var hostName = location.hostname; var url = location.href; 上面几行代码都包含 location 对象.如果使用 with 语句,可以把上面的

  • 深入浅析javascript函数中with

    /*js函数中with函数的用法分析 定义 方便用来引用某个对象中已有的属性 但是不能用来给对象添加属性 要给对象创建 新的属性 必须明确的引用该对象*/ 代码格式 with(object) statements object:新的默认对象 statements:一个或多个语句 oject是该语句的默认对象 with 语句通常用来缩短特定情形下必须写的代码量. x = Math.cos(3 * Math.PI) + Math.sin(Math.LN10) y = Math.tan(14 * Ma

  • javascript之with的使用(阿里云、淘宝使用代码分析)

    记得还在懵懂学习JavaScript基础之时,坊间便有传言"with语句是低效率语句,若非必要,请不要使用该语句",同时, ECMAScript 5 的strict mode下是禁止使用with语句的,所以一直以来我对with语句一直没啥好感. 今天在知乎有个话题大概说的是"你觉得什么东西相当有B格"之类的,然后就有人贴了这段代码: with(document)with(body)with(insertBefore(createElement("scrip

  • Vue js with语句原理及用法解析

    vue源码中编译部分有下面一段代码,里面用到了with: export function generate ( ast: ASTElement | void, options: CompilerOptions ): CodegenResult { const state = new CodegenState(options) const code = ast ? genElement(ast, state) : '_c("div")' return { render: `with(th

  • 由 JavaScript 的 with 引发的探索

    目录 1. 背景 2. with 2.1. with 的性能问题 3. LHS 和 RHS 4. 执行上下文和作用域链 4.1. VO 4.2. AO 4.3. 作用域链 一下文章来源于微信公众号前端巅峰 1. 背景 某天吃饭的时候突然想到,都说 with 会有问题,那么是什么问题,是怎样导致的呢?知其然不知其所以然,在好奇心的驱使下,从 with 出发,一路追溯到 VO.AO.那么先来复习一下 with 是干嘛的吧. 2. with js 的 with 是为对象访问提供命名空间式的访问方式,w

  • javascript 避免闭包引发的问题

    <div id="test"> <div>第一个</div> <div>第二个</div> <div>第三个</div> <div>第四个</div> </div> <script> function test() { var els = document.getElementById("test").getElementsByTagN

  • javascript中length属性的探索

    例子1: 复制代码 代码如下: var obj={0:'a',1:'b'} alert(obj.length); //undefined var arr=['a','b'] alert(arr.length); // 2 从上面的例子看,类数组对象中的length属性并不和它储存的数据数量直接挂钩,无论是索引属性(0,1)还是length属性都作为对象的普通属性存在,它们之间并没有任何关系,js引擎并不会根据储存数据的数量来自动计算类数组对象的长度. 但是类数组对象的length所确实和存储的数

  • react-pdf 打造在线简历生成器的示例代码

    目录 前言 React-PDF简介 程序实现 初始化项目 实现逻辑 遇到问题 重构 部署 参考 前言 PDF 格式是30年前开发的文件格式,并且是使用最广泛的文件格式之一,我们最喜欢使用它作为简历.合同.发票.电子书等文件的格式,最主要的原因是文档格式可以兼容多种设备和应用程序,而且内容 100%保持相同的格式. React-PDF 简介 React PDF 是一个使用 React 创建 PDF 文件的工具,支持在浏览器.移动设备和服务器上创建PDF文件. 可以用它们轻松地将内容呈现到文档中,我

  • JavaScript 沙箱探索

    目录 1.场景 2.沙箱基础功能 3.iframe 实现 4.web worker 实现 5.quickjs 实现 6.结论 1.场景 最近基于 web 在做一些类似于插件系统一样的东西,所以折腾了一下 js 沙箱,以执行第三方应用的代码. 2.沙箱基础功能 在实现之前(好吧,其实是在调研了一些方案之后),确定了沙箱基于 event bus 形式的通信实现上层的功能,基础的接口如下 export interface IEventEmitter { /** * 监听事件 * @param chan

  • javascript开发中因空格引发的错误

    废话不多说,先上代码给大家看看 复制代码 代码如下: <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> <html xmlns="http://www.w3.org/1999/xhtml"> <head> <met

  • javascript prototype的深度探索不是原型继承那么简单第1/3页

    1 什么是prototype JavaScript中对象的prototype属性,可以返回对象类型原型的引用.这是一个相当拗口的解释,要理解它,先要正确理解对象类型(Type)以及原型(prototype)的概念.         前面我们说,对象的类(Class)和对象实例(Instance)之间是一种"创建"关系,因此我们把"类"看作是对象特征的模型化,而对象看作是类特征的具体化,或者说,类(Class)是对象的一个类型(Type).例如,在前面的例子中,p1和

  • javascript中负数算术右移、逻辑右移的奥秘探索

    javascript中负数的算术右移和逻辑右移都十分的让人迷惑,特别是逻辑右移>>>,你会发现即使一个很小的负数,右移之后,也会得到一个无比巨大的数,这是为什么呢? 原来在逻辑右移中符号位会随着整体一起往右移动,这样就是相当于无符号数的移动了,最后得到的就是一个正数,因为符号位不存在了.首先逻辑右移产生的一定是32位的数,然后负数的符号位为1,这意味着从第32位到符号位的位置全部由1填充,这样的数能不大吗例如-1,逻辑右移0位表现形式就是1111 1111 1111 1111 1111

  • 探索Javascript中this的奥秘

    前言: this 是 JavaScript 比较特殊的关键字,运用的地方之广,方式之灵活奠定了它的强大,但同时注定了它的难用 .自己刚开始学的时候被绕的很晕,为了彻底弄懂它查了很多资料.然后将自己学的东西整理了一下,以通俗易懂的方式表达出来,权当做学习笔记,同时也可以给有需要的童鞋做下参考^_^ 什么是this? this 关键字的含义是明确且具体的,即指代当前对象.即意味着这个 this 是在某种相对情况下才成立的. this 被分为三种情况:全局对象.当前对象或者任意对象;判断处于那种情况,

  • 由JavaScript中call()方法引发的对面向对象继承机制call的思考

    起因: 今天在阅读snandy大神的读jQuery之五(取DOM元素)时,看到有讲到toArray()方法,具体jQuery代码如下: 复制代码 代码如下: toArray: function() { return slice.call( this, 0 ); }, get: function( num ) { return num == null ? // Return a 'clean' array this.toArray() : // Return just the object ( n

随机推荐