百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术文章 > 正文

深入理解 JavaScript 中的 with 语句

itomcoil 2025-05-05 16:31 5 浏览

通常来说,所有 JavaScript 开发人员都有一个共同的概念:“避免使用 with 语句。” 这条准则无疑是正确的,但并不是每个人都能很好地解释为什么。虽然只记住“不要使用它”的结果就足够了,但理解其背后的原因对于深入理解 JavaScript 语言和编写高质量代码非常有帮助。

with 语句

with 语句的初衷是为了避免冗长的对象调用:

foo.bar.baz.x = 1;
foo.bar.baz.y = 2;
foo.bar.baz.z = 3;

with(foo.bar.baz) {
    x = 1;
    y = 2;
    z = 3;
}

但实际上,使用变量替换是相当简单的:

var p = foo.bar.baz;
p.x = 1;
p.y = 2;
p.z = 3;

因此,看起来一开始就不需要 with。如今,使用 with 的人已经很少了。在严格模式下,使用 with 会直接报错:

function foo() {
    'use strict';
    with ({}) {}
}

因此,with 已经被完全废弃,人们甚至懒得去关注其原因。

书中的陈述

既然是总结,我想尽可能全面,所以让我们先从书籍开始。关于 JavaScript 的书籍,以下是一些主要参考:

《JavaScript 权威指南》(第 5 版,David Flanagan,P109):

with (Object) statement with 语句用于暂时修改作用域链… 这种语句实际上是将对象添加到作用域链的开头,然后执行语句,再将作用域链恢复到原来的状态… 尽管有时使用 with 语句更方便,但人们反对使用它。使用 with 语句的 JavaScript 代码难以优化,因此其执行速度比不使用 with 语句的等效代码慢得多。此外,在 with 语句中定义函数和初始化变量可能会产生与直觉相悖的意外行为(这种行为及其原因非常复杂,我们在此不再解释)。

《JavaScript 高级程序设计》(第 3 版,Nicholas C. Zakas,P60):

with 语句的目的是将代码的作用域设置为特定对象… 由于频繁使用 with 语句导致的性能下降,以及调试代码的困难,不建议在开发大型应用程序时使用 with 语句。

《JavaScript 语言精粹》(Douglas Crockford,P110):

这个语言中存在 with 语句严重影响了 JavaScript 处理器的速度,因为它破坏了变量名的词法作用域绑定。它的初衷是好的,但如果没有它,JavaScript 语言会稍微好一些。

《深入理解 ECMAScript 6》(Axel Rauschmayer,P153):

这本书是我唯一一本用了一页多的篇幅详细解释了 JavaScript 中废弃 with 的原因的基础参考书。

好了,读了这么多书,让我们现在进入本文的主要话题:

为什么不使用 with 语句?

综上所述,主要考虑如下:

性能问题

with 语句存在明显的性能问题,这在几乎所有参考书中都有提到,但很少有例子来说明这一点。你可以自己进行代码测试,以更直观地量化理解 with 语句的性能。

var a = {a: {a: 1}};
function useWith() {
    with (a.a) {
        for (var i = 0; i < 1000000; i++) {
            a = i; 
        }
    }
}

var b = {b: {b: 1}};
function noWith() {
    for (var i = 0; i < 1000000; i++) {
        b.b.b = i; 
    }
}

var t1 = new Date().getTime();
useWith();
alert(new Date().getTime() - t1);

var t2 = new Date().getTime();
noWith();
alert(new Date().getTime() - t2);

在对象属性赋值一百万次时,性能差异是否显著?

当然,在实际使用中,极少有执行数百万次的循环,损失在可接受范围内。因此,性能损失并不是废弃 with 语句的主要原因。

不可预测性

使用 with 语句导致的不可预测性是废弃 with 的根本原因。with 强行截断词法作用域,临时将对象插入作用域链。这导致代码变得难以捉摸。

例如:

function foo(a) {
    with (a) {
        console.log(a);
    }
}

foo("sword");     // 输出: sword
foo({});          // 输出: [object Object]
foo({a: "sword"}); // 输出: {a: "sword"}

在这个简单例子中,字符串 "sword" 和空对象没有问题。然而,当传递的参数是具有名为 a 的属性的对象时,强行发生 a.a 访问。

这只是一个参数的情况。如果有很多参数呢?当不知道传入参数有什么属性时,可以想象在多个参数之间引用各种属性会有多么混乱。这就是所谓的“令人惊讶和违反直觉”的行为本质。

此外,在 with 语句中声明的变量并不属于 with 指定的对象:

var a = {};   
with (a) {
    x = 'sword';
    var y = 'wang';
}

console.log(a.x);        // undefined
console.log(a.y);        // undefined
console.log(window.x);   // sword
console.log(window.y);   // wang

with 中声明的变量被添加到外部函数中。

function foo() {
    with ({}) { x = 'sword'; }
    console.log(x);
}
foo();  // 输出: sword

这可能和你想象的有些不同。

单单通过标识符及其上下文,是无法确定语句中的标识符指向什么的。这才是 with 被弃用的真正原因。它强行混淆了上下文,使程序的预测和解析变得困难,导致了后面会讨论的优化问题。

代码无法优化

由于无法预测,代码的含义不断变化。不同的调用,甚至相同的调用,由于运行时的变化可能会偏离,使得代码无法优化。

优化涉及两个方面。一方面,解析和执行变慢,这指的是前面提到的性能。另一方面,对于代码优化和压缩工具,如果无法确定是否正在使用变量或属性,则无法重命名(因为属性无法重命名)。

总结

在这个炎热的夏天,我可能被热气蒸得有些思维散乱。心血来潮,我翻出了几本关于 JavaScript 的书,想要探讨一下这个被广泛诟病的 with 语句。说着说着,似乎偏离了主题,胡乱扯了一些看似深奥但不太实用的内容。写完之后,我自己都觉得“哇,这人真闲”。

哦,对了,文章开头还有一个冷笑话,说 JavaScript 比 Java 多 60%。我只是在调侃它们的字符数。“JavaScript” 比 Java 多五个字母;如果你坚持数字符,那么大概是多了 60%。好吧,可能这个笑话有点冷,难怪外面这么热——看来我得冷静一下。

写这些东西可以算是一种消暑和消磨时间的方式。希望你读到这里时,也能在这个夏季找到属于你的凉爽享受。至于 with 语句——了解它并搁置一旁,因为我们反正不会用了,不是吗?

相关推荐

PS小技巧 调整命令,让人物肤色变得更加白皙 #后期修图

我们来看一下如何去将人物的皮肤变得更加的白皙。·首先选中图层,Ctrl键加J键复制一层。·打开这里的属性面板,选择快速操作删除背景,这样就会将人物进行单独的抠取。·接下来在上方去添加一个黑白调整图层,...

把人物肤色提亮的方法和技巧

PS后期调白肤色提亮照片的方法。一白遮百丑,所以对于Photoshop后期来说把人物肤色调白是一项非常重要的任务。就拿这张素材图片来说,这张素材图片人脸的肤色主要偏红、偏黄,也不够白皙,该怎样对它进行...

《Photoshop教程》把美女图片调成清爽色彩及润肤技巧

关注PS精品教程,每天不断更新~~室内人物图片一般会偏暗,人物脸部、肤色及背景会出现一些杂点。处理之前需要认真的给人物磨皮及美白,然后再整体润色。最终效果原图一、用修补工具及图章工具简单去除大一点的黑...

PS后期对皮肤进行美白的技巧

PS后期进行皮肤美白的技巧。PS后期对皮肤进行美白的技巧:·打开素材图片之后直接复制原图。·接下来直接点击上方的图像,选择应用图像命令。·在通道这里直接选择红通道,混合这里直接选择柔光,然后点击确定。...

493 [PS调色]调模特通透肤色

效果对比:效果图吧:1、光位图:2、拍摄参数:·快门:160;光圈:8;ISO:1003、步骤分解图:用曲线调整图层调出基本色调。用可选颜色调整图层调整红色、黄色、白色和灰色4种颜色的混合比例。用色彩...

先选肤色再涂面部,卡戴珊的摄影师透露:为明星拍完照后怎么修图

据英国媒体12月17日报道,真人秀明星金·卡戴珊终于承认,她把女儿小北P进了家族的圣诞贺卡,怪不得粉丝们都表示这张贺卡照得非常失败。上周,这位39岁的女星遭到了一些粉丝针对这张照片的批评,她于当地时间...

如何在PS中运用曲线复制另一张照片的色调

怎样把另一张作品的外观感觉,套用到自己的照片上?单靠肉眼来猜,可能很不容易,而来自BenSecret的教学,关键是在PS使用了两个工具,让你可以准确比较两张照片的曝光、色调与饱和度,方便你调整及复制...

PS在LAB模式下调出水嫩肤色的美女

本PS教程主要使用Photoshop使用LAB模式调出水嫩肤色的美女,教程调色比较独特。作者比较注重图片高光部分的颜色,增加质感及肤色调红润等都是在高光区域完成。尤其在Lab模式下,用高光选区调色后图...

在Photoshop图像后期处理中如何将人物皮肤处理得白皙通透

我们在人像后期处理中,需要将人物皮肤处理的白皙通透,处理方法很多,大多数都喜欢使用曲线、磨皮等进行调整,可以达到亮但是不透,最终效果往往不是很好,今天就教大家一种如何将任务皮肤处理得白皙通透,希望能帮...

PS调色自学教程:宝宝照片快速调通透,简单实用!

PS调色自学教程:宝宝照片快速调通透。·首先复制图层,然后选择进入ACR滤镜,选择曲线锁定照片的亮部,也就高光位置,其他部位补亮一点,尤其是阴影的部位补亮多一些,让画面的层次均匀一点。·然后回到基本项...

【干货】如何利用PS进行人物美化

人物图像美化在Photoshop中非常常用,Photoshop作为一款功能强大的图像处理软件,不仅可以对人像进行基本的调色、美化和修复等处理,还可以改变人物的线条和幅度,如调整脸部器官和脸型的大小、调...

教大家一种可以快速把肤色处理均匀的方法@抖音短视频

快速把肤色处理均匀的方法。今天教大家一种可以快速把肤色处理均匀的方法。像这张照片整体肤色走紫红色,但是局部偏黄缘处理起来非常的麻烦。其实我们只需要新建空白图层,图层混合模式更改为颜色,再选择画笔工具把...

PS调色教程 利用RAW调出干净通透的肤色

要么不发,要么干货。后期教程来噜~用RAW调出干净通透的肤色。这次终于不会原片比PS后好看了吧。如果你依然这么觉得,请不要残忍的告诉我这个事实,泪谢TAT)附送拍摄花絮,感谢各位的支持更多风格请关注m...

photoshop后期皮肤变白的技巧

PS后期皮肤变白的技巧。1.PS后期让皮肤变白的方法有很多种,接下来教你一种非常简单容易上手的方法。2.打开素材图片之后,直接在小太极下拉框的位置添加一个纯色调整图层,颜色设置一个纯白色,点击...

Photoshop调出人物的淡雅粉嫩肤色教程

本教程主要使用Photoshop调出人物的淡雅粉嫩肤色教程,最终的效果非常的通透迷人,下面让我们一起来学习.出自:86ps效果图:原图:1、打开原图复制一层。2、用Topaz滤镜磨皮(点此下载)。3、...