Object.defineProperty() 不詳解

菜菜: “老大玄柠,那個(gè), Object.defineProperty 是什么鬼盒刚?”
假設(shè)我們有個(gè)對(duì)象 user ; 我們要給它增加一個(gè)屬性 name , 我們會(huì)這么做

    var user = {};
    user.name="狂奔的蝸牛";
    console.log(user);//{name: "狂奔的蝸牛"}

如果想要增加一個(gè)sayHi方法叻?

    user.sayHi=function () { console.log("Hi !") };
    console.log(user);//{name: "狂奔的蝸牛", sayHi: ?n}

Object.defineProperty 就是做這個(gè)的

那么Object.defineProperty 怎么用?

Object.defineProperty 需要三個(gè)參數(shù)(object , propName , descriptor)

  • 1 object 對(duì)象 => 給誰加
  • 2 propName 屬性名 => 要加的屬性的名字 【類型:String】
  • 3 descriptor 屬性描述 => 加的這個(gè)屬性有什么樣的特性【類型:Object】

那么descriptor這個(gè)是個(gè)對(duì)象 曾棕,他有那些屬性呢 ? 別著急我們一個(gè)一個(gè)說;
既然可以給一個(gè)對(duì)象增加屬性菜循,那么我們用它來做一下給 user添加 name屬性翘地,代碼是這樣的

    var user = {};
    Object.defineProperty(user,"name",{
        value:"狂奔的蝸牛"
    })
    console.log(user);//{name: "狂奔的蝸牛"}

說明 是的還是那個(gè)經(jīng)典的value屬性,他就是設(shè)置屬性值的癌幕。
等等子眶,屬性值只能為字符串嗎?我們的 number function Object boolean 等呢序芦?

    var user = {};
    Object.defineProperty(user,"name",{
        value:"狂奔的蝸牛"
    })
    Object.defineProperty(user,"isSlow",{
        value:true
    })
    Object.defineProperty(user,"sayHi",{
        value:function () { console.log("Hi !") }
    })
    Object.defineProperty(user,"age",{
        value:12
    })
    Object.defineProperty(user,"birth",{
        value:{
            date:"2018-06-29",
            hour:"15:30"
        }
    })
    console.log(user);

image.png

說明 事實(shí)證明任何類型的數(shù)據(jù)都是可以的哦~

問題又來了臭杰,如果 user對(duì)象已經(jīng)有了name屬性,我們可以通過Object.defineProperty改變這個(gè)值嗎谚中?
我們來試試

    var user = {};
    Object.defineProperty(user,"name",{
        value:"狂奔的蝸牛"
    })
    console.log(user);
    user.name="新=>狂奔的蝸牛"
    console.log(user);

咦渴杆??為什么我改了沒作用勒宪塔?磁奖?
原因:上邊說了descriptor有很多屬性,除了value屬性還有個(gè) writable【顧名思義屬性是否可以被重新賦值】接受數(shù)據(jù)類型為 boolean(默認(rèn)為false) true => 支持被重新賦值 false=>只讀

哦哦某筐,原來如果我沒設(shè)置writable值的時(shí)候就默認(rèn)只讀啊比搭,所以才改不掉

那我們看看,設(shè)置為true南誊,是不是就可以改掉了身诺。

    var user = {};
    Object.defineProperty(user,"name",{
        value:"狂奔的蝸牛",
        writable:true
    })
    console.log(user);
    user.name="新=>狂奔的蝸牛"
    console.log(user);
image.png

這個(gè)descriptor還有其他的屬性嗎?enumerable【顧名思義屬性是否可以被枚舉】接受數(shù)據(jù)類型為 boolean(默認(rèn)為false) true => 支持被枚舉 false=>不支持

額抄囚。霉赡。。枚舉幔托?穴亏?什....什么意思?

假設(shè)我們想知道這個(gè) user對(duì)象有哪些屬性我們一般會(huì)這么做

var user ={
  name:"狂奔的蝸牛",
  age:25
} ;

//es6
var keys=Object.keys(user)
console.log(keys);// ['name','age']
//es5
var keys=[];
for(key in user){
    keys.push(key);
}   
console.log(keys);// ['name','age'] 

如果我們使用 Object.的方式定義屬性會(huì)發(fā)生什么呢重挑?我們來看下輸出

var user ={
  name:"狂奔的蝸牛",
  age:25
} ;
//定義一個(gè)性別 可以被枚舉
Object.defineProperty(user,"gender",{
    value:"男",
    enumerable:true
})

//定義一個(gè)出生日期 不可以被枚舉
Object.defineProperty(user,"birth",{
    value:"1956-05-03",
    enumerable:false
})

//es6
var keys=Object.keys(user)
console.log(keys);
// ["name", "age", "gender"]

console.log(user);
// {name: "狂奔的蝸牛", age: 25, gender: "男", birth: "1956-05-03"}
console.log(user.birth);
// 1956-05-03   

說明 很明顯嗓化,我們定義為 enumerable=falsebirth屬性并沒有被遍歷出來,遍歷 => 其實(shí)就是枚舉(個(gè)人理解啦谬哀,不喜勿噴哦~)
總結(jié) enumerable 屬性取值為 布爾類型 true | false 默認(rèn)值為 false刺覆,為真屬性可以被枚舉;反之則不能玻粪。此設(shè)置不影響屬性的調(diào)用和 查看對(duì)象的值隅津。

configurable 是接下來我們要講的一個(gè)屬性诬垂,這個(gè)屬性有兩個(gè)作用:

  • 1 屬性是否可以被刪除
  • 2 屬性的特性在第一次設(shè)置之后可否被重新定義特性
var user ={
  name:"狂奔的蝸牛",
  age:25
} ;
//定義一個(gè)性別 不可以被刪除和重新定義特性
Object.defineProperty(user,"gender",{
    value:"男",
    enumerable:true,
    configurable:false
})

//刪除一下
delete user.gender;
console.log(user);//{name: "狂奔的蝸牛", age: 25, gender: "男"}

//重新定義特性
Object.defineProperty(user,"gender",{
    value:"男",
    enumerable:true,
    configurable:true
})
// Uncaught TypeError: Cannot redefine property: gender
//會(huì)報(bào)錯(cuò),如下圖
image.png

設(shè)置為 true

var user ={
  name:"狂奔的蝸牛",
  age:25
} ;
//定義一個(gè)性別 可以被刪除和重新定義特性
Object.defineProperty(user,"gender",{
    value:"男",
    enumerable:true,
    configurable:true
})

//刪除前
console.log(user);
// {name: "狂奔的蝸牛", age: 25, gender: "男"}

//刪除一下
delete user.gender;
console.log(user);
// {name: "狂奔的蝸牛", age: 25}

//重新定義特性
Object.defineProperty(user,"gender",{
    value:"男",
    enumerable:true,
    configurable:false
})

//刪除前
console.log(user);
// {name: "狂奔的蝸牛", age: 25, gender: "男"}
//刪除一下 刪除失敗
delete user.gender;
console.log(user);
// {name: "狂奔的蝸牛", age: 25, gender: "男"}

總結(jié) configurable設(shè)置為 true 則該屬性可以被刪除和重新定義特性伦仍;反之屬性是不可以被刪除和重新定義特性的结窘,默認(rèn)值為false(Ps.除了可以給新定義的屬性設(shè)置特性,也可以給已有的屬性設(shè)置特性哈

最后我們來說說充蓝,最重要的兩個(gè)屬性 setget(即存取器描述:定義屬性如何被存人矸恪),這兩個(gè)屬性是做什么用的呢谓苟?我們通過代碼來看看

var user ={
  name:"狂奔的蝸牛"
} ;
var count = 12;
//定義一個(gè)age 獲取值時(shí)返回定義好的變量count
Object.defineProperty(user,"age",{
    get:function(){
        return count;
    }
})
console.log(user.age);//12

//如果我每次獲取的時(shí)候返回count+1呢
var user ={
  name:"狂奔的蝸牛"
} ;
var count = 12;
//定義一個(gè)age 獲取值時(shí)返回定義好的變量count
Object.defineProperty(user,"age",{
    get:function(){
        return count+1;
    }
})
console.log(user.age);//13

接下來我不用解釋了吧官脓,你想在獲取該屬性的時(shí)候?qū)χ底鍪裁措S你咯~

來來來,我們看看 set涝焙,不多說上代碼

var user ={
  name:"狂奔的蝸牛"
} ;
var count = 12;
//定義一個(gè)age 獲取值時(shí)返回定義好的變量count
Object.defineProperty(user,"age",{
    get:function(){
        return count;
    },
    set:function(newVal){
        count=newVal;
    }
})
console.log(user.age);//12
user.age=145;
console.log(user.age);//145
console.log(count);//145

//等等卑笨,如果我想設(shè)置的時(shí)候是 自動(dòng)加1呢?我設(shè)置145 實(shí)際上設(shè)置是146

var user ={
  name:"狂奔的蝸牛"
} ;
var count = 12;
//定義一個(gè)age 獲取值時(shí)返回定義好的變量count
Object.defineProperty(user,"age",{
    get:function(){
        return count;
    },
    set:function(newVal){
        count=newVal+1;
    }
})
console.log(user.age);//12
user.age=145;
console.log(user.age);//146
console.log(count);//146

說明 注意:當(dāng)使用了getter或setter方法仑撞,不允許使用writable和value這兩個(gè)屬性(如果使用赤兴,會(huì)直接報(bào)錯(cuò)滴)
get 是獲取值的時(shí)候的方法,類型為 function 隧哮,獲取值的時(shí)候會(huì)被調(diào)用桶良,不設(shè)置時(shí)為 undefined

set 是設(shè)置值的時(shí)候的方法,類型為 function 沮翔,設(shè)置值的時(shí)候會(huì)被調(diào)用陨帆,undefined

get或set不是必須成對(duì)出現(xiàn),任寫其一就可以

var user ={
  name:"狂奔的蝸牛"
} ;
var count = 12;
//定義一個(gè)age 獲取值時(shí)返回定義好的變量count
Object.defineProperty(user,"age",{
    get:function(){
        console.log("這個(gè)人來獲取值了2墒础疲牵!");
        return count;
    },
    set:function(newVal){
        console.log("這個(gè)人來設(shè)置值了!搏存!");
        count=newVal+1;
    }
})
console.log(user.age);//12
user.age=145;
console.log(user.age);//146
image.png

【完結(jié)】
Object.defineProperty方法直接在一個(gè)對(duì)象上定義一個(gè)新屬性瑰步,或者修改一個(gè)已經(jīng)存在的屬性, 并返回這個(gè)對(duì)象

value: 設(shè)置屬性的值
writable: 值是否可以重寫璧眠。true | false
enumerable: 目標(biāo)屬性是否可以被枚舉。true | false
configurable: 目標(biāo)屬性是否可以被刪除或是否可以再次修改特性 true | false
set: 目標(biāo)屬性設(shè)置值的方法
get:目標(biāo)屬性獲取值的方法

下一篇读虏,我們來看看怎么用它做一個(gè)簡單的雙向綁定
文章傳送門 => 用Object.defineProperty手寫一個(gè)簡單的雙向綁定

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末责静,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子盖桥,更是在濱河造成了極大的恐慌灾螃,老刑警劉巖,帶你破解...
    沈念sama閱讀 222,252評(píng)論 6 516
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件揩徊,死亡現(xiàn)場(chǎng)離奇詭異腰鬼,居然都是意外死亡嵌赠,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,886評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門熄赡,熙熙樓的掌柜王于貴愁眉苦臉地迎上來姜挺,“玉大人,你說我怎么就攤上這事彼硫〈逗溃” “怎么了?”我有些...
    開封第一講書人閱讀 168,814評(píng)論 0 361
  • 文/不壞的土叔 我叫張陵拧篮,是天一觀的道長词渤。 經(jīng)常有香客問我,道長串绩,這世上最難降的妖魔是什么缺虐? 我笑而不...
    開封第一講書人閱讀 59,869評(píng)論 1 299
  • 正文 為了忘掉前任,我火速辦了婚禮礁凡,結(jié)果婚禮上志笼,老公的妹妹穿的比我還像新娘。我一直安慰自己把篓,他們只是感情好纫溃,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,888評(píng)論 6 398
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著韧掩,像睡著了一般紊浩。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上疗锐,一...
    開封第一講書人閱讀 52,475評(píng)論 1 312
  • 那天坊谁,我揣著相機(jī)與錄音,去河邊找鬼滑臊。 笑死口芍,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的雇卷。 我是一名探鬼主播鬓椭,決...
    沈念sama閱讀 41,010評(píng)論 3 422
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼关划!你這毒婦竟也來了小染?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,924評(píng)論 0 277
  • 序言:老撾萬榮一對(duì)情侶失蹤贮折,失蹤者是張志新(化名)和其女友劉穎裤翩,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體调榄,經(jīng)...
    沈念sama閱讀 46,469評(píng)論 1 319
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡踊赠,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,552評(píng)論 3 342
  • 正文 我和宋清朗相戀三年呵扛,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片筐带。...
    茶點(diǎn)故事閱讀 40,680評(píng)論 1 353
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡今穿,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出烫堤,到底是詐尸還是另有隱情荣赶,我是刑警寧澤,帶...
    沈念sama閱讀 36,362評(píng)論 5 351
  • 正文 年R本政府宣布鸽斟,位于F島的核電站拔创,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏富蓄。R本人自食惡果不足惜剩燥,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 42,037評(píng)論 3 335
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望立倍。 院中可真熱鬧灭红,春花似錦、人聲如沸口注。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,519評(píng)論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽寝志。三九已至娇斑,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間材部,已是汗流浹背毫缆。 一陣腳步聲響...
    開封第一講書人閱讀 33,621評(píng)論 1 274
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留乐导,地道東北人苦丁。 一個(gè)月前我還...
    沈念sama閱讀 49,099評(píng)論 3 378
  • 正文 我出身青樓,卻偏偏與公主長得像物臂,于是被迫代替她去往敵國和親旺拉。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,691評(píng)論 2 361

推薦閱讀更多精彩內(nèi)容