js中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ì)象 => 給誰(shuí)加
2 propName 屬性名 => 要加的屬性的名字 【類型:String】
3 descriptor 屬性描述 => 加的這個(gè)屬性有什么樣的特性【類型:Object】
那么descriptor
這個(gè)是個(gè)對(duì)象 ,他有那些屬性呢 ? 別著急我們一個(gè)一個(gè)說(shuō);
既然可以給一個(gè)對(duì)象增加屬性,那么我們用它來(lái)做一下給 user添加 name屬性,代碼是這樣的
var user = {}; Object.defineProperty(user,"name",{ value:"狂奔的蝸牛" }) console.log(user);//{name: "狂奔的蝸牛"}
說(shuō)明 是的還是那個(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);
說(shuō)明 事實(shí)證明任何類型的數(shù)據(jù)都是可以的哦~
問(wèn)題又來(lái)了,如果 user對(duì)象已經(jīng)有了name屬性,我們可以通過(guò)Object.defineProperty改變這個(gè)值嗎?
我們來(lái)試試
var user = {}; Object.defineProperty(user,"name",{ value:"狂奔的蝸牛" }) console.log(user); user.name="新=>狂奔的蝸牛" console.log(user);
咦??為什么我改了沒(méi)作用勒??
原因:上邊說(shuō)了descriptor有很多屬性,除了value屬性還有個(gè) writable【顧名思義屬性是否可以被重新賦值】接受數(shù)據(jù)類型為 boolean(默認(rèn)為false) true => 支持被重新賦值 false=>只讀
哦哦,原來(lái)如果我沒(méi)設(shè)置writable值的時(shí)候就默認(rèn)只讀啊,所以才改不掉
那我們看看,設(shè)置為true,是不是就可以改掉了。
var user = {}; Object.defineProperty(user,"name",{ value:"狂奔的蝸牛", writable:true }) console.log(user); user.name="新=>狂奔的蝸牛" console.log(user);
這個(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ā)生什么呢?我們來(lái)看下輸出
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
說(shuō)明 很明顯,我們定義為 enumerable=false
的birth
屬性并沒(méi)有被遍歷出來(lái),遍歷 => 其實(shí)就是枚舉(個(gè)人理解啦,不喜勿噴哦~)
總結(jié) enumerable
屬性取值為 布爾類型 true | false
默認(rèn)值為 false
,為真屬性可以被枚舉;反之則不能。此設(shè)置不影響屬性的調(diào)用和 查看對(duì)象的值。
configurable
是接下來(lái)我們要講的一個(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ò),如下圖
設(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è)置特性哈
)
最后我們來(lái)說(shuō)說(shuō),最重要的兩個(gè)屬性 set
和get
(即存取器描述:定義屬性如何被存?。@兩個(gè)屬性是做什么用的呢?我們通過(guò)代碼來(lái)看看
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
接下來(lái)我不用解釋了吧,你想在獲取該屬性的時(shí)候?qū)χ底鍪裁措S你咯~
來(lái)來(lái)來(lái),我們看看 set,不多說(shuō)上代碼
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
說(shuō)明 注意:當(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),任寫(xiě)其一就可以
var user ={ name:"狂奔的蝸牛" } ; var count = 12; //定義一個(gè)age 獲取值時(shí)返回定義好的變量count Object.defineProperty(user,"age",{ get:function(){ console.log("這個(gè)人來(lái)獲取值了!!"); return count; }, set:function(newVal){ console.log("這個(gè)人來(lái)設(shè)置值了?。?); count=newVal+1; } }) console.log(user.age);//12 user.age=145; console.log(user.age);//146
【完結(jié)】
Object.defineProperty
方法直接在一個(gè)對(duì)象上定義一個(gè)新屬性,或者修改一個(gè)已經(jīng)存在的屬性, 并返回這個(gè)對(duì)象
- value: 設(shè)置屬性的值
- writable: 值是否可以重寫(xiě)。true | false
- enumerable: 目標(biāo)屬性是否可以被枚舉。true | false
- configurable: 目標(biāo)屬性是否可以被刪除或是否可以再次修改特性 true | false
- set: 目標(biāo)屬性設(shè)置值的方法
- get:目標(biāo)屬性獲取值的方法
下一篇,我們來(lái)看看怎么用它做一個(gè)簡(jiǎn)單的雙向綁定
文章傳送門 => 用Object.defineProperty手寫(xiě)一個(gè)簡(jiǎn)單的雙向綁定
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
JS實(shí)現(xiàn)復(fù)制粘貼文字及圖片功能
Clipboard?API?是一組用于在瀏覽器中操作剪貼板的?JavaScript?API,它允許開(kāi)發(fā)者在網(wǎng)頁(yè)上讀取和寫(xiě)入剪貼板內(nèi)容,實(shí)現(xiàn)復(fù)制、剪切和粘貼等功能,這篇文章主要介紹了JS實(shí)現(xiàn)復(fù)制粘貼文字及圖片功能,需要的朋友可以參考下2024-07-07JQuery,Extjs,YUI,Prototype,Dojo 等JS框架的區(qū)別和應(yīng)用場(chǎng)景簡(jiǎn)述
隨著web2.0的彪悍發(fā)展,以及瀏覽器端所承載的工作越來(lái)越大(在不是很影響性能的情況下,開(kāi)發(fā)者都習(xí)慣把能用瀏覽器做的事兒都讓瀏覽器做,以減輕服務(wù)器的壓力和帶寬費(fèi)用等)。2010-04-04JavaScript表格隔行變色和Tab標(biāo)簽頁(yè)特效示例【附j(luò)Query版】
這篇文章主要介紹了JavaScript表格隔行變色和Tab標(biāo)簽頁(yè)特效,結(jié)合實(shí)例形式分析了javascript元素遍歷、事件響應(yīng)相關(guān)操作技巧,并附帶jQuery對(duì)應(yīng)實(shí)現(xiàn)代碼供大家參考,需要的朋友可以參考下2019-07-07this,this,再次討論javascript中的this,超全面(經(jīng)典)
在JavaScript中,this 的概念比較復(fù)雜。除了在面向?qū)ο缶幊讨?,this 還是隨處可用的。這篇文章介紹了javascript中的this相關(guān)知識(shí),對(duì)javascript this相關(guān)知識(shí)感興趣的朋友一起學(xué)習(xí)吧2016-01-01基于javascript實(shí)現(xiàn)句子翻牌網(wǎng)頁(yè)版小游戲
這篇文章主要介紹了基于javascript實(shí)現(xiàn)句子翻牌網(wǎng)頁(yè)版小游戲的相關(guān)資料,需要的朋友可以參考下2016-03-03JavaScript 中Date對(duì)象的格式化代碼方法匯總
JavaScript默認(rèn)的時(shí)間格式我們一般情況下不會(huì)用,所以需要進(jìn)行格式化,下面小編給大家?guī)?lái)了三種js date對(duì)象格式化實(shí)例代碼,需要的朋友參考下吧2017-09-09