delete
操作符用于删除对象的某个属性。
语法
delete expression
expression 的计算机结果应该是某个属性的引用,例如:
delete object.property delete object['property']
参数
object
- 对象的名称,或计算结果为对象的表达式。
property
- 要删除的属性。
返回值
对于所有情况都是true,除非属性是一个自己的不可配置属性,在这种情况下,非严格模式返回 false。
异常
在严格模式下,如果是对象自身的不可配置属性,会抛出Global_objects/SyntaxError
。
描述
与通常对 delete 的理解不同,delete 操作符与直接释放内存无关。内存管理 通过断开引用来间接完成的,查看内存管理页可了解详情。
delete
操作符会从某个对象上移除指定属性。成功删除的时候回返回 true
,否则返回 false
。但是,重要的是要考虑以下情况:
- 如果你删除的属性在对象上不存在,那么delete将不会起作用,但仍会返回true
- 如果
delete
操作符删除成功,则被删除的属性将从所属的对象上彻底消失。然后,如果该对象的原型链上有一个同名属性,则该对象会从原型链上继承该同名属性。(也就是说delete操作只会在自身的属性上起作用) - 任何使用
var
声明的属性不能从全局作用域或函数的作用域中删除。- 这样的话,delete操作不能删除任何在全局作用域中的函数(无论这个函数是来自于函数声明或函数表达式)
- 除了在全局作用域中的函数不能被删除,在对象(object)中的函数是能够用delete操作删除的。
- 任何用
let
或const
声明的属性不能够从它被声明的作用域中删除。 - 不可设置的(Non-configurable)属性不能被移除。这意味着像
Math
,Array
,Object
内置对象的属性以及使用Object.defineProperty()
方法设置为不可设置的属性不能被删除。
下面的代码块给出了一个简单的例子:
var Employee = { age: 28, name: 'abc', designation: 'developer' } console.log(delete Employee.name); // returns true console.log(delete Employee.age); // returns true // 当试着删除一个不存在的属性时 // 同样会返回true console.log(delete Employee.salary); // returns true
不可配置属性
当一个属性被设置为不可设置,delete操作将不会有任何效果,并且会返回false。在严格模式下会抛出语法错误(SyntaxError
)。
var Employee = {}; Object.defineProperty(Employee, 'name', {configurable: false}); console.log(delete Employee.name); // returns false
var
, let
以及const
创建的不可设置的属性不能被delete操作删除。
var nameOther = 'XYZ'; // We can access this global property using: Object.getOwnPropertyDescriptor(window, 'nameOther'); // output: Object {value: "XYZ", // writable: true, // enumerable: true, // configurable: false} // 因为“nameOther”使用var关键词添加, // 它被设置为不可设置(non-configurable) delete nameOther; // return false
在严格模式下,这样的操作会抛出异常。
严格模式与非严格模式的对比
在严格模式下,如果直接对一个变量的引用、函数的参数或者函数名使用delete操作,将会抛出语法错误(SyntaxError
)。
任何使用var声明的变量都会被标记为不可设置的。在下面的例子中,salary是不可设置的以及不能被删除的。在非严格模式下,下面的delete操作将会返回false。
function Employee() { delete salary; var salary; } Employee();
让我们来看看相同的代码在严格模式下会有怎样的表现。会抛出一个语法错误( SyntaxError)而不是返回false。
"use strict"; function Employee() { delete salary; // SyntaxError var salary; } // 相似的,任何对任何函数 // 直接使用delete操作将会抛出语法错误。 function DemoFunction() { //some code } delete DemoFunction; // SyntaxError
示例
// 在全局作用域创建 adminName 属性 adminName = 'xyz'; // 在全局作用域创建 empCount 属性 // 因为我们使用了 var,它会标记为不可配置。如果使用 let 或 const 而会是可配置的。 var empCount = 43; EmployeeDetails = { name: 'xyz', age: 5, designation: 'Developer' }; // adminName 是全局作用域的一个属性。 // 因为它不是用 var 创建的,所在可以删除。 // 因此,它是可配置的。 delete adminName; // 返回 true // 相反,empCount 是不可配置的, // 因为创建它时使用了 var。 delete empCount; // 返回 false // delete 可用于删除对象的属性 delete EmployeeDetails.name; // 返回 true // 甚至属性不存在,它也会返回 "true" delete EmployeeDetails.salary; // 返回 true // delete 对内建静态属性不起作用 delete Math.PI; // 返回 false // EmployeeDetails 是全局作用域的一个属性。 // 因为定义它的时候没有使用 "var",它被标记为可配置。 delete EmployeeDetails; // 返回 true function f() { var z = 44; // delete 对局部变量名不起作用 delete z; // 返回 false }
delete 和原型链
在下面的示例中,我们删除一个对象的自己的属性,而具有相同名称的属性在原型链上可用:
function Foo() { this.bar = 10; } Foo.prototype.bar = 42; var foo = new Foo(); // 返回 true,因为删除的是 foo 对象的自身属性 delete foo.bar; // foo.bar 仍然可用,因为它在原型链上可用。 console.log(foo.bar); // 从原型上删除属性 delete Foo.prototype.bar; // 输出 "undefined",因为不能继承这个属性了 console.log(foo.bar);
删除数组元素
当你删除一个数组元素时,数组的 length 属性并不会变小。即便你删除了数组的最后一个元素也是如此。
当用 delete
操作符删除一个数组元素时,被删除的元素已经不再属于该数组。下面的例子中用 delete
删除了 trees[3]
。
var trees = ["redwood","bay","cedar","oak","maple"]; delete trees[3]; if (3 in trees) { // 这里不会执行 }
如果你想让一个数组元素继续存在但是其值是 undefined
,那么可以使用将 undefined
赋值给这个元素而不是使用 delete
。下面的例子中,trees[3] 被赋值为 undefined
,但该元素仍然存在。
var trees = ["redwood","bay","cedar","oak","maple"]; trees[3] = undefined; if (3 in trees) { // 这里会被执行 }
规范
规范 | 状态 | 备注 |
---|---|---|
ECMAScript Latest Draft (ECMA-262) The delete Operator |
Living Standard | |
ECMAScript 2015 (6th Edition, ECMA-262) The delete Operator |
Standard | |
ECMAScript 5.1 (ECMA-262) The delete Operator |
Standard | |
ECMAScript 1st Edition (ECMA-262) The delete Operator |
Standard | Initial definition. Implemented in JavaScript 1.2. |
浏览器兼容性
特性 | Chrome | Firefox (Gecko) | Internet Explorer | Opera | Safari |
---|---|---|---|---|---|
Basic support | (Yes) | (Yes) | (Yes) | (Yes) | (Yes) |
Temporal dead zone | ? | 36 (36) | ? | ? | ? |
Feature | Android | Chrome for Android | Firefox Mobile (Gecko) | IE Mobile | Opera Mobile | Safari Mobile |
---|---|---|---|---|---|---|
Basic support | (Yes) | (Yes) | (Yes) | (Yes) | (Yes) | (Yes) |
Temporal dead zone | ? | ? | 36.0 (36) | ? | ? | ? |
浏览器兼容性
虽然 ECMAScript 规定了对象的遍历顺序是由对象定义时属性的书写顺序决定的。(译者注:ES5已经对遍历机制做了调整,重新规定:属性遍历的顺序是没有被规定的), 大部分浏览器都依照这个规定,先添加的属性先被遍历(除了从原型上继承的属性)(译者注:Chrome和Opera已经遵循了ES5的新规定,具体请看)。但是,在 Internet Explorer 中,使用 delete
删除一个属性后,奇怪的事情发生了,如果被删除的属性重新被添加,那么遍历时,该属性的顺序会是上次删除前的那个位置所应该有的顺序,而不是出现在遍历的最后一个。
所以,如果你想让对象的遍历顺序兼容所有的浏览器,那么你可以使用两个数组来模拟 (一个做为keys,一个做为 values), 或者建立一个由单一属性的对象组成的数组,等。
相关链接
<embed height="0" id="xunlei_com_thunder_helper_plugin_d462f475-c18e-46be-bd10-327458d045bd" type="application/thunder_download_plugin" width="0">