js中的內部屬性與delete操做符

本文正式地址:http://www.xiabingbao.com/javascript/2015/08/03/javascript-delete-configurablejavascript

在講解Configurable以前,咱們首先來看一道面試題:java

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。chrome

除了上述的兩種狀況,還有其餘的各類經常使用變量也有能被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是獲取形參的個數:this

function func1(a, b){
}
console.log( func1.length );        // 2
console.log( delete func1.length ); // true,刪除成功
console.log( func1.length );        // 0

刪除經常使用變量:prototype

console.log( delete NaN );      // false,刪除失敗
console.log( delete undefined );// false
console.log( delete Infinity ); // false
console.log( delete null );     // true,刪除成功

刪除prototype,而不是刪除prototype上的屬性:code

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-configurable

相關文章
相關標籤/搜索