解析JavaScript中delete操作符不能刪除的對(duì)象
ES3 中,delete在8.6.2.5及11.4.1有介紹,如下
有一些信息,
1、實(shí)現(xiàn)上delete操作符會(huì)調(diào)用引擎內(nèi)部的[[Delete]]方法
2、[[Delete]]在8.6.2里定義
3、刪除的屬性有個(gè)DontDelete的特性,如果有,delete時(shí)直接返回false
搜索“DontDelete”,會(huì)發(fā)現(xiàn)有很多,如下都不能delete
1, 激活對(duì)象的arguments對(duì)象 (10.1.6)
function func() {
delete arguments;
alert(arguments);
}
func(1);
2,變量聲明 (10.2.1)
var a = 10;
delete a;
alert(a); // 10
這一條在很多JS書里有提及,即不能delete掉使用var聲明的變量。
3,函數(shù)聲明
function func() {}
delete func;
alert(func); // func code
4,函數(shù)的length屬性
function func(a, b) {}
delete func.length;
alert(func.length); // 2
5,一些常量(NaN、Infinity、undefined)
delete NaN; // false
delete Infinity; // false
delete undefined; // false
6,內(nèi)置構(gòu)造器的prototype
delete Object.prototype; // false
delete Function.prototype; // false
delete Array.prototype; // false
delete ExpReg.prototype; // false
delete Date.prototype; // false
delete Error.prototype; // false
delete Number.prototype; // false
delete Boolean.prototype; // false
delete String.prototype; // false
7, 數(shù)組和字符串的length
var arr = [], str = 'hello';
delete arr.length; // false
delete str.length; // false
8,Math對(duì)象的屬性(Math.E、Math.LN10、Math.LN2、Math.LOG2E、Math.LOG10E、Math.PI、Math.SQRT1_2、Math.SQRT2)
delete Math.E; // false
...
9,正則對(duì)象的屬性(source、global、ignoreCase、multiline、lastIndex)
var reg = /ss/;
delete reg.source; // false
...
ES5 與ES3不同,ES5中沒(méi)有“DontDelete”,卻增加了 [[Configurable]] (8.6.1)。
如果該值為false,則不能delete,以上列舉的9點(diǎn)在ES5中描述為[[Configurable]]為false。
ES5新增的Object.defineProperty方法可顯示的定義對(duì)象的Configurable,如下
var obj = {name: 'John'};
Object.defineProperty(obj, "key", {
configurable: false,
value: "static"
});
delete obj.name; // true
delete obj.key // false
對(duì)象obj有name,key。name可以delete,key則不行。
此外ES5嚴(yán)格模式中delete configuable為false的對(duì)象時(shí)會(huì)直接拋異常。如
"use strict";
delete Object.prototype;
FF中控制臺(tái)報(bào)錯(cuò)如下
除了內(nèi)置對(duì)象的一些方法或?qū)傩圆荒軇h除外,自定義對(duì)象也有不能刪除的。如delete不能刪除對(duì)象繼承來(lái)自原型上的屬性
function Person() {}
Person.prototype.name = 'John Backus';
var p = new Person();
delete p.name;
console.log(p.name); // 仍然輸出 John Backus
如果this和prototype上都有name,那么delete后,會(huì)將prototype上的呈現(xiàn)出來(lái)
function Person() {
this.name = 'John Backus';
}
Person.prototype.name = 'John Resig';
var p = new Person();
console.log(p.name); // John Backus
delete p.name;
console.log(p.name); // John Resig, 來(lái)自原型
如果非要?jiǎng)h除原型上的name,只能
delete Person.prototype.name
總結(jié)下:
1,內(nèi)置對(duì)象的屬性及方法多數(shù)不能delete(雖然有些能delete,如isNaN、parseInt)
2,對(duì)象繼承于原型的屬性和方法不能delete
原因也很簡(jiǎn)單,
1,內(nèi)置對(duì)象的屬性及方法多數(shù)不能delete保護(hù)該語(yǔ)言最核心API,這些API被delete了,基本上就廢了。如delete Object.prototype。
2,對(duì)象繼承于原型的屬性和方法不能delete是出于保護(hù)原型,否則 “類A的對(duì)象delete了原型上的屬性,那么繼承于A的都將丟失該屬性”。
相關(guān)文章
JavaScript Tips 使用DocumentFragment加快DOM渲染速度
大家在開發(fā)JavaScript應(yīng)用的時(shí)候,如果遇到這種大量節(jié)點(diǎn)的情況,不妨將DocumentFragment作為一個(gè)備選的方案。2010-06-06在Html中使用Requirejs進(jìn)行模塊化開發(fā)實(shí)例詳解
在前端模塊化的時(shí)候,不僅僅是js需要進(jìn)行模塊化管理,html有時(shí)候也需要模塊化管理。這里就介紹下如何通過(guò)requirejs,實(shí)現(xiàn)html代碼的模塊化開發(fā)2016-04-04JS時(shí)間戳與日期格式的轉(zhuǎn)換小結(jié)
這篇文章主要介紹了JS時(shí)間戳與日期格式的轉(zhuǎn)換小結(jié),本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友參考下吧2024-01-01關(guān)于axios返回空對(duì)象的問(wèn)題解決
這篇文章主要給大家介紹了關(guān)于axios返回空對(duì)象的問(wèn)題解決方法,文中介紹的非常詳細(xì),相信對(duì)大家學(xué)習(xí)或者使用axios具有一定的參考價(jià)值,需要的朋友們下面來(lái)一起看看吧。2017-04-04js點(diǎn)亮星星評(píng)分并獲取參數(shù)的js代碼
點(diǎn)亮星星評(píng)分后,點(diǎn)擊按鈕,立即獲得分?jǐn)?shù)參數(shù)值,方便不想使用ajax傳參的朋友2014-07-07基于JS實(shí)現(xiàn)限時(shí)搶購(gòu)倒計(jì)時(shí)間表代碼
本文給大家分享一段簡(jiǎn)單的代碼基于js實(shí)現(xiàn)限時(shí)搶購(gòu)倒計(jì)時(shí)間表功能,非常不錯(cuò),代碼簡(jiǎn)單易懂,需要的的朋友參考下吧2017-05-05layui 點(diǎn)擊重置按鈕, select 并沒(méi)有被重置的解決方法
今天小編就為大家分享一篇layui 點(diǎn)擊重置按鈕, select 并沒(méi)有被重置的解決方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2019-09-09