JavaScript物件導向詳細解析之屬性描述符

2022-05-27 18:01:25
本篇文章給大家帶來了關於的相關知識,其中主要介紹了關於物件導向的相關問題,包括了屬性描述符、資料描述符、存取描述符等等內容,下面一起來看一下,希望對大家有幫助。

【相關推薦:、】

JavaScript物件導向——屬性描述符

JavaScript其實支出多種程式設計正規化的,包括函數語言程式設計和物件導向程式設計:

  • JavaScript中的物件被設計成一組 屬性的無序集合,像是一個嘻哈表,有key和value組成;
  • key是一個識別符號名稱,value可以是任意型別,也可以是其他物件或者函數型別;
  • 如果值是一個函數,那麼我們可以稱之為物件的方法

1、如何建立一個物件呢?

  • 早起使用建立物件方式最多的是使用Object類,並且使用new關鍵字來建立一個物件,然後再將屬性或方法存入物件中:
var obj = new Object()
obj.name = 'why'
console.log(obj.name, obj) // why { name: 'why' }
  • 後來很多開發者為了方便起見,都是直接通過字面量的形式來建立物件:
// 字面量方式
var obj2 = { name: 'jam', age: '8' }
console.log(obj) // { name: 'jam', age: '8' }

2、操作物件的屬性——屬性描述符

在之前我們的屬性都是直接定義在物件內部,或者直接新增到物件內部;
但是這樣我們就不能對這個屬性進行一些限制:比如這個屬性是否可以通過 delect刪除,是否可以在 for-in遍歷的時候被遍歷出來呢?
如果我們想要對一個屬性進行比較精準的操作控制,那麼我就可以使用 屬性描述符。通過屬性描述符可以精準的新增或修改物件的屬性;
屬性描述符需要使用 Object.defineProperty來對屬性進行新增或修改。

屬性描述符分為兩種:資料描述符和存取描述符

2.1資料描述符

資料描述符是一個具有值的屬性,該值可能是可寫的,也可能是不可寫的。資料描述符具有以下可選鍵值:

  • value:該屬性對應的值。可以是任何有效的JavaScript值(數值,物件,函數等)。預設為undefined。
  • writable:當且僅當該屬性的writable為true時,value才能被複制運運算元改變。預設為false。
  • configurable:當且僅當該屬性的configurable為true時,該屬性描述符才能夠被改變,同時該屬性也能從對應的物件上被刪除。預設為false。
  • enumerable:當且僅當該屬性的enumerable為true時,該屬性才能夠出現在物件的列舉屬性中。預設為false。

2.2.1、獲取屬性描述符 Object.getOwnPropertyDescriptor()

Object.getOwnPropertyDescriptor()方法返回指定物件上一個自有屬性對應的屬性描述符。

Object.getOwnPropertyDescriptor(obj,prop)
  • obj:需要查詢的目標物件
  • prop:目標物件內屬性名稱(String型別)。
  • 返回值:如果指定的屬性存在在於物件上,則返回其屬性描述符物件,否則返回undefined。

注意:如果該方法的第一個引數不是物件,會報錯(TypeError)。

2.1.2、設定屬性描述符 Object.defineProperty

Object.defineProperty() 方法會直接在一個物件上定義一個新屬性,或者修改一個物件的現有屬性, 並返回這個物件。

Object.defineProperty(obj,prop,descriptor)
  • obj:要在其上定義屬性的物件。
  • prop:要定義或修改的屬性的名稱。
  • descriptor:將被定義或修改的屬性描述符
  • 返回值:被傳遞給函數的物件
如下範例程式碼展示了屬性描述符的設定和獲取
var obj = {
    name: 'jam',
    age: 8
}
Object.defineProperty(obj, 'job', {
    value: '律師'
})
console.log(Object.getOwnPropertyDescriptor(obj, 'age')) // { value: 8, writable: true, enumerable: true, configurable: true }

console.log(obj.job) // 律師
// 通過defineProperty新增的屬性,該新屬性是不可修改、不可刪除以及不可列舉的
console.log(Object.getOwnPropertyDescriptor(obj, 'job')) // {value: '律師',writable: false,enumerable: false,configurable: false}

注意:通過defineProperty新增的屬性,該新屬性是不可修改、不可刪除以及不可列舉的

(1)configurable在物件上是否可刪除

var obj = {
    name: 'jam',
    age: 8
}
Object.defineProperty(obj, 'address', {
    value: '河北',
    // configurable 該屬性不可刪除,更不可再次使用defineProperty修改屬性描述符
    configurable: false,

})
delete obj.address // 想使用delete刪除該屬性
obj.address = '廣州' // 想修改obj中的屬性address值為廣州
console.log(obj.address)  // 輸出結果:河北

因為屬性描述符configurable值為false 不可刪除和修改,所以delete和修改都沒生效

(2)enumerable 是否可列舉遍歷

var obj = {
    name: 'jam',
    age: 8}Object.defineProperty(obj, 'sex ', {
    value: '男',
    // enumerable 設定該屬性是否可以列舉
    enumerable: true})for (i in obj) {
    console.log(i)}

當enumerable: false時,輸出結果為 name age
當enumerable: true時,輸出結果為 name age sex

(3)writable 該特性控制該屬性是否可以賦值(寫入值)

var obj = {
    name: 'jam',
    age: 8}Object.defineProperty(obj, 'score', {
    value: 80,
    // writable: true 
    writable: false})obj.score = 100 console.log(obj.score) // 80

因為writeable的值為false,所以修改score為100時,並沒有修改成功,最終輸出80

2.1.3、同時設定多個屬性描述符 Object.defineProperties

是不是感覺每次只能設定一個屬性的屬性描述符很繁瑣,Object.defineProperties可以幫你解決問題。
Object.defineProperties()方法為物件定義一個或多個新屬性或修改現有屬性,並返回該物件。

Object.defineProperties(obj,props)
  • obj:要在其上定義屬性的物件。
  • props:要定義其可列舉屬性或修改的屬性描述符的物件。
  • 返回值:被傳遞給函數的物件。

範例程式碼如下:

var obj = {
    name: 'jam',}Object.defineProperties(obj, {
    'age': {
        value: 28,
        writable: true,
        configurable: false,
        enumerable: true
    },
    'job': {
        value: '律師',
        writable: true,
        configurable: false,
        enumerable: true
    },
    'sex': {
        value: '男',
        writable: false,
        configurable: false,
        enumerable: true
    },
    'height': {
        value: '1.8 m',
        writable: false,
        configurable: false,
        enumerable: true
    }})console.log(obj) //  name: 'jam', age: 28, job: '律師', sex: '男', height: '1.8m' }

2.2存取描述符

存取描述符是由getter-setter函數對描述的屬性。存取描述符具有以下可選鍵值:

  • get:給屬性提供getter的方法,如果沒有getter則為undefined。當存取該屬性時,該方法會被執行,方法執行時沒有引數傳入,但是會傳入this物件。
  • set:給屬性提供setter的方法,如果沒有setter則為undefined。當屬性值修改時,觸發執行該方法。該方法將接受唯一引數,即該屬性新的引數值。
  • configurable:當且僅當該屬性的configurable為true時,該屬性描述符才能夠被改變,同時該屬性也能從對應的物件上被刪除。預設為false。
  • enurnerable:當且僅當該屬性的enurnerable為true時,該屬性才能夠出現在物件的列舉屬性中。預設為false。

configurable 和 enurnerable 的使用與資料描述符中的一致,這裡就不過多講解了。
主要講一下get 和 set 方法的使用

2.2.1 get()與set()的使用

var obj = {
    name: 'jam',
    age: 8,
    _address: '河北'
}

// 存取描述符的使用場景
// 1.隱藏某一個私有屬性被希望直接被外界使用和賦值
// 2.如果我們希望解惑某一個屬性它存取和設定值的過程時,也會使用儲存屬性描述符
Object.defineProperty(obj, 'address', {
    enumerable: true,
    configurable: true,
    get: function () {
        foo()
        return this._address
    },
    set: function (value) {
        bar()
        this._address = value
    }
})
function foo () {
    console.log("截獲了一次address的值")
}

function bar () {
    console.log("設定了一次address的值")
}

上述範例程式碼控制檯列印結果如下:
在這裡插入圖片描述

【相關推薦:、】

以上就是JavaScript物件導向詳細解析之屬性描述符的詳細內容,更多請關注TW511.COM其它相關文章!