JS 中的内部属性与 delete 操作符

本文正式地点:http://www.xiabingbao.com/javascript/2015/08/03/javascript-delete-conf…

在解说Configurable之前,我们起首来看一道面试题:

a = 1;
console.log( window.a ); // 1
console.log( delete window.a ); // true
console.log( window.a ); // undefined

var b = 2;
console.log( window.b ); // 2
console.log( delete window.b ); // false
console.log( window.b ); // 2

从上面的这道题可以看出两个的区分:在没有运用var声明变量时,运用delete关键词是可以举行删除的,再次猎取时价就是undefined了;在运用var声明的变量,运用delete是不能删除的,再猎取时价依然是2。

1. delete 操纵符

运用delete删除变量或属性时,删除胜利返回true,不然返回false。如上面的例子中,delete没法删除变量a时,则返回false;而delete能胜利删除变量b,则返回true。

除了上述的两种状况,另有其他的种种经常使用变量也有能被delete删除的,也有不能被删除的。我们先不论delete这些变量时,为何会发生如许的效果,这里只看他的返回值:

删除delete数组中个中的一个元素:

// 运用for~in是轮回不到的,直接疏忽到该元素
// 运用for()可以获得该元素,然则值是undefined
var arr = [1, 2, 3, 4];
console.log( arr );             // [1, 2, 3, 4]
console.log( delete arr[2] );   // true,删除胜利
console.log( arr );             // [1, 2, undefined, 4]

删除function范例的变量:

// chrome 不能删除;火狐可以删除
function func(){
}
console.log( func );
console.log( delete func );
console.log( func );

删除function.length,该length是猎取形参的个数:

function func1(a, b){
}
console.log( func1.length );        // 2
console.log( delete func1.length ); // true,删除胜利
console.log( func1.length );        // 0

删除经常使用变量:

console.log( delete NaN );      // false,删除失利
console.log( delete undefined );// false
console.log( delete Infinity ); // false
console.log( delete null );     // true,删除胜利

删除prototype,而不是删除prototype上的属性:

function Person(){
}
Person.prototype.name = "蚊子";
console.log( delete Person.prototype ); // false,没法删除
console.log( delete Object.prototype ); // false

删除数组和字符串的length时:

var arr = [1, 2, 3, 4];
console.log( arr.length );          // 4
console.log( delete arr.length );   // false,删除失利
console.log( arr.length );          // 4

var str = 'abcdefg';
console.log( str.length );          // 7
console.log( delete str.length );   // false,删除失利
console.log( str.length );          // 7

删除obj中的属性时:

var obj = {name:'wenzi', age:25};
console.log( obj.name );        // wenzi
console.log( delete obj.name ); // true,删除胜利
console.log( obj.name );        // undefined
console.log( obj );             // { age:25 }

删除实例对象中的属性时,从以下的输出效果可以看出,运用delete删除属性时,删除的仅仅是实例对象自身的属性,而不能删除prototype上的属性,纵然再删一次也是删撤除不的;若要删除prototype上的属性的属性或要领,只能是:delete Person.prototype.name

function Person(){
    this.name = 'wenzi';
}
Person.prototype.name = '蚊子';
var student = new Person();
console.log( student.name );        // wenzi
console.log( delete student.name ); // true,删除胜利
console.log( student.name );        // 蚊子
console.log( delete student.name ); // true
console.log( student.name );        // 蚊子
console.log( delete Person.prototype.name );// true,删除胜利
console.log( student.name );        // undefined

2. js 的内部属性

在上面的例子中,有的变量或属性可以删除胜利,而有的变量或属性则没法举行删除,那是什么决议这个变量或属性能不能被删除呢。

ECMA-262第5版定义了JS对象属性中特征(用于JS引擎,外部没法直接接见)。ECMAScript中有两种属性:数据属性和接见器属性。

2.1 数据属性

数据属性指包含一个数据值的位置,可在该位置读取或写入值,该属性有4个供述其行动的特征:

  • [[configurable]]:示意可否运用delete操纵符删除从而从新定义,或可否修正成接见器属性。默以为true;
  • [[Enumberable]]:示意是不是可经由过程for-in轮回返回属性。默许true;
  • [[Writable]]:示意是不是可修正属性的值。默许true;
  • [[Value]]:包含该属性的数据值。读取/写入都是该值。默以为undefined;如上面实例对象Person中定义了name属性,其值为’wenzi’,对该值的修正都横竖在这个位置

要修正对象属性的默许特征(默许都为true),可挪用Object.defineProperty()要领,它吸收三个参数:属性地点对象,属性名和一个描述符对象(必需是:configurable、enumberable、writable和value,可设置一个或多个值)。

以下:

var person = {};
Object.defineProperty(person, 'name', {
    configurable: false,    // 不可删除,且不能修正成接见器属性
    writable: false,        // 不可修正
    value: 'wenzi'          // name的值为wenzi
});
console.log( person.name);          // wenzi
console.log( delete person.name );  // false,没法删除
person.name = 'lily';
console.log( person.name );         // wenzi

可以看出,delete及重置person.name的值都没有见效,这就是因为挪用defineProperty函数修正了对象属性的特征;值得注重的是一旦将configurable设置为false,则没法再运用defineProperty将其修正成true(实行会报错:Uncaught TypeError: Cannot redefine property: name);

2.2 接见器属性

它重要包含一对getter和setter函数,在读取接见器属性时,会挪用getter返回有效值;写入接见器属性时,挪用setter,写入新值;该属性有以下4个特征:

  • [[Configurable]]:是不是可经由过程delete操纵符删除从新定义属性;
  • [[Numberable]]:是不是可经由过程for-in轮回查找该属性;
  • [[Get]]:读取属性时自动挪用,默许:undefined;
  • [[Set]]:写入属性时自动挪用,默许:undefined;

接见器属性不能直接定义,必需运用defineProperty()来定义,以下:

var person = {
    _age: 18
};
Object.defineProperty(person, 'isAdult', {
    Configurable : false,
    get: function () {
        if (this._age >= 18) {
            return true;
        } else {
            return false;
        }
    }
});
console.log( person.isAdult );  // true

不过照样有一点须要分外注重一下,Object.defineProperty()要领设置属性时,不能同时声明接见器属性(set和get)和数据属性(writable或许value)。意义就是,某个属性设置了writable或许value属性,那末这个属性就不能声明get和set了,反之亦然。

如若像下面的体式格局举行定义,接见器属性和数据属性同时存在:

var o = {};
Object.defineProperty(o, 'name', {
    value: 'wenzi',
    set: function(name) {
        myName = name;
    },
    get: function() {
        return myName;
    }
});

上面的代码看起来貌似是没有什么问题,然则真正实行时会报错,报错以下:

Uncaught TypeError: Invalid property. A property cannot both have accessors and be writable or have a value

关于数据属性,可以获得:configurable,enumberable,writable和value;

关于接见器属性,可以获得:configurable,enumberable,get和set。

由此我们可知:一个变量或属性是不是可以被删除,是由其内部属性Configurable举行掌握的,若Configurable为true,则该变量或属性可以被删除,不然不能被删除。

但是我们应当怎样猎取这个Configurable值呢,总不能用delete尝尝能不能删除吧。有方法滴!!

2.3 猎取内部属性

ES5为我们供应了Object.getOwnPropertyDescriptor(object, property)来猎取内部属性。

如:

var person = {name:'wenzi'};
var desp = Object.getOwnPropertyDescriptor(person, 'name'); // person中的name属性
console.log( desp );    // {value: "wenzi", writable: true, enumerable: true, configurable: true}

经由过程Object.getOwnPropertyDescriptor(object, property)我们可以猎取到4个内部属性,configurable掌握着变量或属性是不是可被删除。这个例子中,person.name的configurable是true,则申明是可以被删除的:

console.log( person.name );         // wenzi
console.log( delete person.name );  // true,删除胜利
console.log( person.name );         // undefined

我们再回到最最先的谁人面试题:

a = 1;
var desp = Object.getOwnPropertyDescriptor(window, 'a');
console.log( desp.configurable );   // true,可以删除

var b = 2;
var desp = Object.getOwnPropertyDescriptor(window, 'b');
console.log( desp.configurable );   // false,不能删除

跟我们运用delete操纵删除变量时发生的效果是一样的。

3. 总结

别看一个简简单单的delete操纵,内里实在包含了许多的道理!

本文正式地点:http://www.xiabingbao.com/javascript/2015/08/03/javascript-delete-conf…

    原文作者:wenzi
    原文地址: https://segmentfault.com/a/1190000003048399
    本文转自网络文章,转载此文章仅为分享知识,如有侵权,请联系博主进行删除。
点赞