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

來源:互聯網
上載者:User

js中的內部屬性與delete操作符
在講解Configurable之前,我們首先來看一道面試題: 

a = 1;console.log( window.a ); // 1console.log( delete window.a ); // falseconsole.log( window.a ); // 1var b = 2;console.log( window.b ); // 2console.log( delete window.b ); // trueconsole.log( window.b ); // undefined

 

從上面的這道題可以看出兩個的區別:在沒有使用var聲明變數時,使用delete關鍵詞是不能進行刪除的,依然能擷取到變數a的值;在使用var聲明的變數,就能使用delete刪除,再擷取時值就是undefined了。 1. delete操作符使用delete刪除變數或屬性時,刪除成功返回true,否則返回false。如上面的例子中,delete無法刪除變數a時,則返回false;而delete能成功刪除變數b,則返回true。 除了上述的兩種情況,還有其他的各種常用變數也有能被delete刪除的,也有不能被刪除的。我們先不管delete這些變數時,為什麼會產生這樣的結果,這裡只看他的傳回值: 刪除delete數組中其中的一個元素: 
// 使用for~in是迴圈不到的,直接忽略到該元素// 使用for()可以得到該元素,但是值是undefinedvar 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 不能刪除;Firefox可以刪除function func(){}console.log( func );console.log( delete func );console.log( func );

 

刪除function.length,該length是擷取形參的個數: 
function func1(a, b){}console.log( func1.length );      // 2console.log( delete func1.length ); // true,刪除成功console.log( func1.length );       // 0

 

刪除常用變數: 
console.log( delete NaN );     // false,刪除失敗console.log( delete undefined );// falseconsole.log( delete Infinity ); // falseconsole.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 );         // 4console.log( delete arr.length );  // false,刪除失敗console.log( arr.length );         // 4var str = 'abcdefg';console.log( str.length );         // 7console.log( delete str.length );  // false,刪除失敗console.log( str.length );         // 7

 

刪除obj中的屬性時:
var obj = {name:'wenzi', age:25};console.log( obj.name );       // wenziconsole.log( delete obj.name ); // true,刪除成功console.log( obj.name );       // undefinedconsole.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 );        // wenziconsole.log( delete student.name ); // true,刪除成功console.log( student.name );        // 蚊子console.log( delete student.name ); // trueconsole.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個供述其行為的特性: 要修改對象屬性的預設特徵(預設都為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); // wenziconsole.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個特徵: 訪問器屬性不能直接定義,必須使用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 ); // wenziconsole.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操作,裡面其實包含了很多的原理!

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.