不推荐使用with语句,因为可能造成bug或者性能损失。详见"Description"的"Ambiguity Con"部分。

概要

with语句的作用是扩展作用域链(scope chain)。

语法

with (expression) {
  statement
}
expression
将expression添加到作用域链,以便在statement运行时使用。expression 外面需加括号。
statement
任何语句都行。如果不止一句,则需用block符号({ ... })将其括起來。

说明

JavaScript查找某个未使用命名空间的变量时,会通过作用域链来查找,作用域链是跟执行代码的context或者包含这个变量的函数有关。'with'语句將某个对象添加的作用域链的顶部,如果在statement中有某个未使用命名空间的变量,跟作用域链中的某个属性同名,则这个变量将指向这个属性值。如果沒有同名的属性,则将拋出ReferenceError异常。

不推荐使用with,在ECMAScript 5 strict mode中该标签已被禁止。 推荐的替代方案是声明一个临时变量来承载你所需要的属性。

性能方面的利与弊

利:with語句可以在不造成性能損失的情況下,減少變量的長度。其造成的附加計算量很少。使用'with'可以減少不必要的指針路徑解析運算。需要注意的是,很多情況下,也可以不使用with語句,而是使用一個臨時變量來保存指針,來達到同樣的效果。

弊:with語句使得程序在查找變量值時,都是先在指定的對象中查找。所以那些本來不是這個對象的屬性的變量,查找起來將會很慢。如果是在對性能要求較高的場合,'with'下面的statement語句中的變量,只應該包含這個指定對象的屬性。

语义不明的弊端

弊端:with语句使得代码不易阅读,同时使得JavaScript编译器不易在作用域链上查找某个变量,不易決定应该在哪个对象上來取值。请看下面的例子:

function f(x, o) {
  with (o) 
    print(x);
}

f被调用时,x有可能能取到值,也可能是undefined,如果能取到, 有可能是在o上取的值,也可能是函數的第一個參數x的值(如果o中沒有這個屬性的話)。如果你忘記在作為第二個參數的對象o中定義x這個屬性,程序並不會報錯,只是取到另一個值而已。

弊端:使用with語句的代碼,無法向前兼容,特別是在使用一些原生數據類型的時候。看下面的例子:

function f(foo, values) {
    with (foo) {
        console.log(values)
    }
}

如果是在ECMAScript 5環境調用f([1,2,3], obj),則with語句中變量values將指向函數的第二個參數values。但是,ECMAScript 6標準給Array.prototype添加了一個新屬性values,所有數組實例將繼承這個屬性。所以在ECMAScript 6環境中,with語句中變量values將指向[1,2,3].values

示例

Example: Using with

下面的with語句指定Math对象作为默认对象with语句里面的变量,分別指向Math对象的PI 、cossin函數,不用在前面添加命名空间(Math.PI)

var a, x, y;
var r = 10;

with (Math) {
  a = PI * r * r;
  x = r * cos(PI);
  y = r * sin(PI / 2);
}

技术规范

Specification Status Comment
ECMAScript 1st Edition. Standard Initial definition.
ECMAScript 5.1 (ECMA-262)
with statement
Standard Now forbidden in strict mode.
ECMAScript 2015 (6th Edition, ECMA-262)
with statement
Standard  

浏览器兼容性

Feature Chrome Firefox (Gecko) Internet Explorer Opera Safari
Basic support (Yes) (Yes) (Yes) (Yes) (Yes)
Feature Android Chrome for Android Firefox Mobile (Gecko) IE Mobile Opera Mobile Safari Mobile
Basic support (Yes) (Yes) (Yes) (Yes) (Yes) (Yes)

另见

文档标签和贡献者

 此页面的贡献者: wizardforcel, YFM-getA, jonkee, SphinxKnight, front
 最后编辑者: wizardforcel,