Vue源碼學(xué)習(xí)defineProperty響應(yīng)式數(shù)據(jù)原理實(shí)現(xiàn)
準(zhǔn)備工作
接上文數(shù)據(jù)初始化完成之后,就可以對(duì)數(shù)據(jù)進(jìn)行劫持。Vue2中對(duì)數(shù)據(jù)進(jìn)行劫持采用了一個(gè)Api叫Object.defineProperty()
在這里需要提供一個(gè)方法去觀測(cè)data變化,這個(gè)方法是一個(gè)核心模塊(響應(yīng)式模塊),我們單獨(dú)建一個(gè)文件夾來(lái)存放在/src/observe/index.js
// src/state.js import { observe } from "./observe/index" export function initState(vm){ // 對(duì)數(shù)據(jù)需要進(jìn)行劫持 const opts = vm.$options //獲取所有選項(xiàng) if (opts.data){ initData(vm) } } function initData(vm){ // 對(duì)數(shù)據(jù)進(jìn)行代理 let data = vm.$options.data // data可以是函數(shù)或者對(duì)象,根實(shí)例可以是對(duì)象,組件data必須是函數(shù) data = typeof data === 'function' ? data.call(vm) : data // 對(duì)數(shù)據(jù)進(jìn)行劫持 Vue2采用的一個(gè)api object.defineproperty observe(data) }
// src/observe/index.js export function observe(data){ debugger console.log(data); }
執(zhí)行/dist/index.html,當(dāng)控制臺(tái)出可以輸出{name: 'i東東', age: 18}
說(shuō)明前面的代碼沒(méi)有問(wèn)題,接下來(lái)就可以開(kāi)始下面的操作了。
第一步 對(duì)對(duì)象進(jìn)行劫持
當(dāng)拿到了data,就可以對(duì)data數(shù)據(jù)進(jìn)行劫持,如果說(shuō)他不是對(duì)象就不用劫持,所以還需要進(jìn)行一個(gè)判斷。
// src/observe/index.js export function observe(data){ // 對(duì)這個(gè)對(duì)象進(jìn)行劫持 if(typeof data !=='object'|| data == null){ return // 只對(duì)對(duì)象進(jìn)行劫持 } }
那么緊接著如何劫持這個(gè)對(duì)象呢?
如果一個(gè)對(duì)象被劫持過(guò)了,那么就不需要再次被劫持了(要判斷一個(gè)對(duì)象是否被劫持過(guò),可以增添一個(gè)實(shí)例,來(lái)判斷是否被劫持過(guò)),所以在內(nèi)部創(chuàng)造了一個(gè)類(lèi)去觀測(cè)數(shù)據(jù),如果數(shù)據(jù)被觀測(cè)過(guò)那他的實(shí)例就是這個(gè)類(lèi)。
// src/observe/index.js class Observer{ constructor(data){ //所有數(shù)據(jù) this.walk(data) // 因?yàn)閐ata是一個(gè)對(duì)象,所以就需要對(duì)data進(jìn)行比遍歷 } walk(data){ // 循環(huán)對(duì)象 對(duì)屬性依次劫持 Object.keys(data).forEach(key=>defineReactive(data,key,data[key])) //重新定義屬性 } } export function defineReactive(target,key,value){ // 閉包 屬性劫持 Object.defineProperty(target,key,{ get(){ //取值的時(shí)候會(huì)執(zhí)行g(shù)et return value }, set(newValue){ // 修改的時(shí)候執(zhí)行set if(newValue === value) return value = newValue } }) } export function observe(data){ // 對(duì)這個(gè)對(duì)象進(jìn)行劫持 if(typeof data !=='object'|| data == null){ return // 只對(duì)對(duì)象進(jìn)行劫持 } return new Observer(data); // 對(duì)這個(gè)數(shù)據(jù)進(jìn)行觀測(cè) }
因?yàn)橐獙?duì)每個(gè)屬性進(jìn)行劫持,但是Objece.defineProperty()
只能劫持已經(jīng)存在的屬性,后增加的或者刪除的是不知道的,(Vue2里面會(huì)為此單獨(dú)寫(xiě)一些api 比如:setset setdelete),所以需要對(duì)data進(jìn)行遍歷 this.walk()
對(duì)屬性依次劫持,重新定義屬性(性能會(huì)差,Vue3中proxy就會(huì)好很多),就可以調(diào)用defineReactive,因?yàn)檫@個(gè)方法可以單獨(dú)去使用,所以直接導(dǎo)出。
完成之后執(zhí)行index.html中console.log(vm),會(huì)發(fā)現(xiàn)vm上只有用戶的選項(xiàng),并沒(méi)有剛才劫持過(guò)的屬性,是因?yàn)樵趕tate.js中我們只是data傳入了observe函數(shù),所以就考慮,在vm上增加一個(gè)屬性,叫_data
,這樣就相當(dāng)于把_data對(duì)象放在了實(shí)例vm上,并且又把這個(gè)對(duì)象進(jìn)行了觀測(cè),觀測(cè)的時(shí)候依舊回去循環(huán)這個(gè)對(duì)象。
// src/state.js function initData(vm){ let data = vm.$options.data data = typeof data === 'function' ? data.call(vm) : data vm._data = data // 新增這一句 observe(data) }
這樣再次輸出,會(huì)發(fā)現(xiàn)控制臺(tái)輸出了_data,并且給age,name都增加上來(lái)get和set方法,現(xiàn)在說(shuō)明這個(gè)事情就成了。
這個(gè)時(shí)候就可以通過(guò)vm._data.name進(jìn)行取值
// dist/index.html const vm = new Vue({ data(){ return { name:'i東東', age:18 } } }) vm._data.name = 'i東東修改' console.log(vm._data.name); // 用戶設(shè)置值了 // index.js:15 用戶取值了 // index.html:29 i東東修改
第二步 修改取值方法
緊接著就會(huì)發(fā)現(xiàn)正常我們?nèi)≈刀际莢m.name,但是上面的訪問(wèn)還是vm._data.name,所以下面需要將取值的方法進(jìn)行一下優(yōu)化。需要在state.js中將vm._data用vm代理。
// state.js function proxy(vm,target,key){ Object.defineProperty(vm,key,{ get(){ return vm[target][key] // vm._data.name }, set(newValue){ vm[target][key] = newValue } }) } function initData(vm){ // 對(duì)數(shù)據(jù)進(jìn)行代理 let data = vm.$options.data data = typeof data === 'function' ? data.call(vm) : data vm._data = data observe(data) // 新增 將vm._data用vm代理 for(let key in data){ proxy(vm,'_data',key) } }
這樣在index.html中我們就可以用過(guò)vm.name重鋼訪問(wèn)到數(shù)據(jù),也可以通過(guò)vm.name = 'i東東修改'去設(shè)置值,雖然這樣性能是不太好的,但是他用起來(lái)會(huì)很方便的。所以在這里面相當(dāng)于代理了兩次第一次把用戶的數(shù)據(jù)進(jìn)行了屬性劫持,第二次就是proxy當(dāng)取值和設(shè)置值的時(shí)候代理到某個(gè)人身上。
第三步 深度屬性劫持
// index.html const vm = new Vue({ data(){ return { name:'i東東', age:18, say:{ hobby:'學(xué)習(xí)' } } } }) console.log(vm);
假如說(shuō)我再增加一個(gè)對(duì)象say,輸出vm會(huì)發(fā)現(xiàn)hobby并沒(méi)有被劫持,原因是因?yàn)槲覀冎唤俪至薾ame、age、say三個(gè)屬性,如果屬性是個(gè)對(duì)象的話,我們就需要再次劫持。這樣我們只需要在defineReactive()里面再次調(diào)用observe再次建立劫持,形成遞歸這樣就可以完成對(duì)對(duì)象的深度屬性劫持。
// src/observe/index.js export function defineReactive(target,key,value){ // 閉包 屬性劫持 observe(value) // 新增 對(duì)所有的對(duì)象都進(jìn)行屬性接觸 Object.defineProperty(target,key,{ get(){ //取值的時(shí)候會(huì)執(zhí)行g(shù)et console.log('用戶取值了'); return value }, set(newValue){ // 修改的時(shí)候執(zhí)行set console.log('用戶設(shè)置值了'); if(newValue === value) return value = newValue } }) }
以上就是Vue源碼學(xué)習(xí)defineProperty響應(yīng)式數(shù)據(jù)原理實(shí)現(xiàn)的詳細(xì)內(nèi)容,更多關(guān)于Vue defineProperty響應(yīng)式數(shù)據(jù)的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
關(guān)于SpringBoot與Vue交互跨域問(wèn)題解決方案
最近在利用springboot+vue整合開(kāi)發(fā)一個(gè)前后端分離的個(gè)人博客網(wǎng)站,所以這一篇總結(jié)一下在開(kāi)發(fā)中遇到的一個(gè)問(wèn)題,關(guān)于解決在使用vue和springboot在開(kāi)發(fā)前后端分離的項(xiàng)目時(shí),如何解決跨域問(wèn)題。在這里分別分享兩種方法,分別在前端vue中解決和在后臺(tái)springboot中解決。2021-10-10vue+elementUI中使用Echarts之餅圖問(wèn)題
這篇文章主要介紹了vue+elementUI中使用Echarts之餅圖問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-10-10Vue.js中用v-bind綁定class的注意事項(xiàng)
關(guān)于數(shù)據(jù)綁定一個(gè)常見(jiàn)需求就是操作元素的class列表和它的內(nèi)聯(lián)樣式。因?yàn)樗鼈兌际菍傩?,我們可以?v-bind 處理它們,但是使用v-bind綁定class的時(shí)候我們要有一些注意事項(xiàng),下面這篇文章就給大家分享了下要注意的方面,希望能對(duì)大家有所幫助,下面來(lái)一起看看吧。2016-12-12使用Vue3+ts?開(kāi)發(fā)ProTable源碼教程示例
這篇文章主要為大家介紹了使用Vue3+ts?開(kāi)發(fā)ProTable源碼示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-07-07vue-editor-bridge報(bào)錯(cuò)的解決方案
這篇文章主要介紹了vue-editor-bridge報(bào)錯(cuò)的解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-04-04vue實(shí)現(xiàn)頁(yè)面div盒子拖拽排序功能
本文主要介紹了vue實(shí)現(xiàn)頁(yè)面div盒子拖拽排序功能,文中通過(guò)示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-10-10