iis服务器助手广告
返回顶部
首页 > 资讯 > 精选 >由JavaScript的with 引发的探索是怎样的
  • 566
分享到

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

2023-06-26 04:06:37 566人浏览 泡泡鱼
摘要

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

本篇文章给大家分享的是有关由javascript的with 引发的探索是怎样的,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。

1. 背景

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

2. with

jswith 是为对象访问提供命名空间式的访问方式,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 引发的探索是怎样的,小编相信有部分知识点可能是我们日常工作会见到或用到的。希望你能通过这篇文章学到更多知识。更多详情敬请关注编程网精选频道。

--结束END--

本文标题: 由JavaScript的with 引发的探索是怎样的

本文链接: https://www.lsjlt.com/news/306574.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

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

下载Word文档
猜你喜欢
  • 由JavaScript的with 引发的探索是怎样的
    本篇文章给大家分享的是有关由JavaScript的with 引发的探索是怎样的,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。1. 背景某天吃饭的时候突然想到,都说 w...
    99+
    2023-06-26
  • 由 JavaScript 的 with 引发的探索
    目录1. 背景2. with2.1. with 的性能问题3. LHS 和 RHS4. 执行上下文和作用域链4.1. VO4.2. AO4.3. 作用域链一下文章来源于微信公众号前端...
    99+
    2024-04-02
  • MongoDB复合索引引发的灾难是怎样的
    这期内容当中小编将会给大家带来有关MongoDB复合索引引发的灾难是怎样的,文章内容丰富且以专业的角度为大家分析和叙述,阅读完这篇文章希望大家可以有所收获。前情提要11月末我司商品服务的MongoDB主库曾...
    99+
    2024-04-02
  • MongoDB中的索引是怎样的
    本篇文章给大家分享的是有关MongoDB中的索引是怎样的,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。索引就像图书的目录一样,可以让我们快速定位到需要的内容,关系型数据库中有索...
    99+
    2023-06-19
  • 关于mysql的索引是怎样的
    本篇文章给大家分享的是有关关于mysql的索引是怎样的,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。一、mysql中的索引是什么?索引在mys...
    99+
    2024-04-02
  • MySQL索引结构是怎么样的
    这篇文章主要为大家展示了“MySQL索引结构是怎么样的”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“MySQL索引结构是怎么样的”这篇文章吧。数据库存储单位首先...
    99+
    2024-04-02
  • javascript是由谁开发的
    这篇文章主要为大家展示了“javascript是由谁开发的”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“javascript是由谁开发的”这篇文章吧。javascript是Netscape公司开...
    99+
    2023-06-15
  • 基于Java的全文索引引擎Lucene是怎样的
    今天给大家介绍一下基于Java的全文索引引擎Lucene是怎样的。文章的内容小编觉得不错,现在给大家分享一下,觉得有需要的朋友可以了解一下,希望对大家有所帮助,下面跟着小编的思路一起来阅读吧。基于Java的全文索引引擎LuceneLucen...
    99+
    2023-06-03
  • ORA-00130 引发的故障是怎样的
    本篇文章为大家展示了ORA-00130 引发的故障是怎样的,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。 1.登录系统检查服务  &nb...
    99+
    2024-04-02
  • MySQL的覆盖索引与回表是怎样的
    今天就跟大家聊聊有关MySQL的覆盖索引与回表是怎样的,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。两大类索引使用的存储引擎:MySQL5.7 In...
    99+
    2024-04-02
  • mongodb中索引分类是怎样的以及如何创建索引
    mongodb中索引分类是怎样的以及如何创建索引,相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。学习索引分类和创建索引:   &nbs...
    99+
    2024-04-02
  • Node和JavaScript中的模板引擎是怎样的
    本篇文章给大家分享的是有关Node/JavaScript中的模板引擎是怎样的,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。简介我们将介绍如何用...
    99+
    2024-04-02
  • Node.js V8 引擎与 WebAssembly:探索 JavaScript 的新兴前沿
    Node.js 是一个基于 JavaScript 的运行时环境,而 V8 是 Chrome 浏览器中用于执行 JavaScript 代码的高性能引擎。WebAssembly 是一种二进制指令格式,旨在在 Web 浏览器和诸如 Node.j...
    99+
    2024-04-02
  • SQLSERVER聚集索引和主键的误区是怎样的
    这期内容当中小编将会给大家带来有关SQLSERVER聚集索引和主键的误区是怎样的,文章内容丰富且以专业的角度为大家分析和叙述,阅读完这篇文章希望大家可以有所收获。很多人会把Primary Key和聚集索引搞...
    99+
    2024-04-02
  • MySQL 8.0新特性倒叙索引是怎样的
    这篇文章将为大家详细讲解有关MySQL 8.0新特性倒叙索引是怎样的,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文章后对相关知识有一定的了解。Oracle使用desc index的场景...
    99+
    2024-04-02
  • JavaScript引入模块的历史简介是怎样的
    JavaScript引入模块的历史简介是怎样的,很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。随着我们的应用越来越大,我们想要将...
    99+
    2024-04-02
  • Git索引文件的结构是什么样的?
    Git是一个开源的分布式版本控制系统,它可以帮助程序员更好地管理代码。在Git中,索引文件(也称为缓存或暂存区)是一个非常重要的概念。它是一个二进制文件,用于记录暂存区的状态,以便在提交新的版本时使用。 本文将深入探讨Git索引文件的结构...
    99+
    2023-05-26
    javascript git 索引
  • JavaScript中实现new的两种方式引发的探究
    前言 当你 new 一个构造函数时发生了什么? “众所周知”的三步: 创建一个空对象,将它的引用赋给 this,继承函数的原型;通过 this 将属性和方法添加至这个对象;最后返回 ...
    99+
    2024-04-02
  • MySQL和Oracle中的唯一性索引的差别是怎样的
    本篇文章给大家分享的是有关MySQL和Oracle中的唯一性索引的差别是怎样的,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。  &...
    99+
    2024-04-02
  • MySQL的索引原理以及查询优化是怎样的
    本篇文章为大家展示了MySQL的索引原理以及查询优化是怎样的,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。一、介绍1.什么是索引?一般的应用系统,读写比例在10:1左右,而且插入操作和一般的更新操作...
    99+
    2023-06-22
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作