欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

3分鐘了解vue數(shù)據(jù)劫持的原理實(shí)現(xiàn)

 更新時(shí)間:2019年05月01日 15:02:37   作者:海明月  
這篇文章主要介紹了3分鐘了解vue數(shù)據(jù)劫持的原理實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧

目的: 了解Object.defineProperty如何實(shí)現(xiàn)數(shù)據(jù)劫持

大致原理是這樣的:

  1. 定義一個(gè)監(jiān)聽函數(shù),對對象的每一個(gè)屬性進(jìn)行監(jiān)聽
  2. 通過Object.defineProperty對監(jiān)聽的每一個(gè)屬性設(shè)置get 和 set 方法。
  3. 對對象實(shí)行監(jiān)聽
  4. 對對象內(nèi)嵌對象進(jìn)行處理
  5. 對數(shù)組對象進(jìn)行處理

1. 先定義一個(gè)對象

let obj = {
 name: 'jw'
}

2. 定義一個(gè)監(jiān)聽函數(shù)

/**
* 判斷監(jiān)聽的是否是對象
* 如果是對象,就遍歷,并且對每個(gè)屬性進(jìn)行定義get 和 set
*/

function observer(obj) {
 if(typeof obj === 'object') {
  for (let key in obj) {
  // defineReactive 方法設(shè)置get和set,見第三步
   defineReactive(obj, key, obj[key]);
  }
 }
}

3.定義一個(gè)函數(shù),處理每個(gè)屬性

function defineReactive(obj, key, value) {
 Object.defineProperty(obj, key, {
  get() {
   return value;
  },
  set(val) {
   console.log('數(shù)據(jù)更新了')
   value = val;
  }
 })
}

ok. 到這里初版已經(jīng)實(shí)現(xiàn)了。嘗試一下吧

observer(obj);
obj.name = 'haha'

控制臺輸出:
//數(shù)據(jù)更新了

以上已經(jīng)實(shí)現(xiàn)設(shè)置obj的屬性的時(shí)候,被監(jiān)聽到,并且可以去執(zhí)行一些代碼了。但是,如果對象里面嵌入了對象呢?比如:

let obj = {
 name: 'jw',
 age: {
  old: 60
 }
}

執(zhí)行以下代碼

observer(obj);
obj.age.old = '50'

控制臺輸出: 空

4.對監(jiān)控的obj進(jìn)行迭代處理

// 修改defineReactive , 添加一行代碼
function defineReactive(obj, key, value) {
 // 如果對象的屬性也是一個(gè)對象。迭代處理
 observer(value);
 Object.defineProperty(obj, key, {
  //....
 })
}

再執(zhí)行以下代碼:

observer(obj);
obj.age.old = '50'

控制臺輸出:
//數(shù)據(jù)更新了

可惜的是,如果對象是一個(gè)數(shù)組,Object.defineProperty就無法起作用了,比如:

obj.skill = [1, 2, 3];
obj.age.push(4);

控制臺輸出:
//空

實(shí)際上,不止push,包括slice,shift,unshif...都是沒有作用.

5. 重寫數(shù)組的方法

let arr = ['push', 'slice', 'shift', 'unshift'];
arr.forEach(method=> {
 let oldPush = Array.prototype[method];
 Array.prototype[method] = function(value) {
  console.log('數(shù)據(jù)更新了')
  oldPush.call(this, value)
 }
})

再執(zhí)行以下代碼:

obj.skill = [1, 2, 3];
obj.skill.push(4);

控制臺輸出:
//數(shù)據(jù)更新了

但是,數(shù)組的length操作仍然是無效的。這也是為什么vue中只能通過方法去改變數(shù)組的原因了。

總結(jié): Object.defineProperty只是解決了狀態(tài)變更后,如何觸發(fā)通知的問題,那要通知誰呢?誰會關(guān)心那些屬性發(fā)生了變化呢?以后再說。

以下完整代碼

let obj = {
 name: 'jw',
 age: {
  old: '60'
 }
}

// vue 數(shù)據(jù)劫持 Observer.defineProperty

function observer(obj) {
 if(typeof obj === 'object') {
  for (let key in obj) {
   defineReactive(obj, key, obj[key]);
  }
 }
}

function defineReactive(obj, key, value) {
 observer(value);

 Object.defineProperty(obj, key, {
  get() {
   return value;
  },
  set(val) {
   console.log('數(shù)據(jù)更新了')
   value = val;
  }
 })
}
observer(obj);


// obj.age.old = '50'


// Object.defineProperty 對 數(shù)組無效
let arr = ['push', 'slice', 'shift', 'unshift'];

arr.forEach(method=> {
 let oldPush = Array.prototype[method];
 Array.prototype[method] = function(value) {
  console.log('數(shù)據(jù)更新了')
  oldPush.call(this, value)
 }
})
obj.skill = [1, 2, 3];
obj.skill.push(4);

以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

  • vue3.0?setup中使用vue-router問題

    vue3.0?setup中使用vue-router問題

    這篇文章主要介紹了vue3.0?setup中使用vue-router問題,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-10-10
  • Vue實(shí)現(xiàn)省市區(qū)三級聯(lián)動(dòng)

    Vue實(shí)現(xiàn)省市區(qū)三級聯(lián)動(dòng)

    這篇文章主要為大家詳細(xì)介紹了Vue實(shí)現(xiàn)省市區(qū)三級聯(lián)動(dòng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2020-12-12
  • ElementUI多個(gè)子組件表單的校驗(yàn)管理實(shí)現(xiàn)

    ElementUI多個(gè)子組件表單的校驗(yàn)管理實(shí)現(xiàn)

    這篇文章主要介紹了ElementUI多個(gè)子組件表單實(shí)現(xiàn)方法,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-11-11
  • vue關(guān)于錨點(diǎn)定位、跳轉(zhuǎn)到指定位置實(shí)現(xiàn)方式

    vue關(guān)于錨點(diǎn)定位、跳轉(zhuǎn)到指定位置實(shí)現(xiàn)方式

    這篇文章主要介紹了vue關(guān)于錨點(diǎn)定位、跳轉(zhuǎn)到指定位置實(shí)現(xiàn)方式,具有很好的參考價(jià)值,希望對大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2024-04-04
  • vue封裝公共方法的實(shí)現(xiàn)代碼

    vue封裝公共方法的實(shí)現(xiàn)代碼

    這篇文章給大家介紹了vue封裝公共方法的實(shí)現(xiàn),文章中通過代碼示例給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作有一定的幫助,需要的朋友可以參考下
    2024-01-01
  • 基于Vue實(shí)現(xiàn)的多條件篩選功能的詳解(類似京東和淘寶功能)

    基于Vue實(shí)現(xiàn)的多條件篩選功能的詳解(類似京東和淘寶功能)

    這篇文章主要介紹了Vue多條件篩選功能,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-05-05
  • Vue學(xué)習(xí)-VueRouter路由基礎(chǔ)

    Vue學(xué)習(xí)-VueRouter路由基礎(chǔ)

    這篇文章主要介紹了Vue學(xué)習(xí)-VueRouter路由基礎(chǔ),路由本質(zhì)上就是超鏈接,xiamian?文章圍繞VueRouter路由的相關(guān)資料展開詳細(xì)內(nèi)容,需要的小伙伴可以參考一下,希望對你的學(xué)習(xí)有所幫助
    2021-12-12
  • Vue單文件組件基礎(chǔ)模板小結(jié)

    Vue單文件組件基礎(chǔ)模板小結(jié)

    本篇文章主要介紹了Vue單文件組件基礎(chǔ)模板小結(jié),小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2017-08-08
  • Vue前端導(dǎo)出Excel文件的詳細(xì)實(shí)現(xiàn)方案

    Vue前端導(dǎo)出Excel文件的詳細(xì)實(shí)現(xiàn)方案

    在開發(fā)后臺管理系統(tǒng)的時(shí)候,很多地方都要用到導(dǎo)出excel表格,比如將table中的數(shù)據(jù)導(dǎo)出到本地,下面這篇文章主要給大家介紹了關(guān)于Vue導(dǎo)出Excel文件的相關(guān)資料,需要的朋友可以參考下
    2021-09-09
  • Vue路由管理器Vue-router的使用方法詳解

    Vue路由管理器Vue-router的使用方法詳解

    本文將詳細(xì)介紹Vue路由管理器Vue-router的使用方法詳解,需要的朋友可以參考下
    2020-02-02

最新評論