亚洲激情专区-91九色丨porny丨老师-久久久久久久女国产乱让韩-国产精品午夜小视频观看

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

使用vue實現數據劫持的原理是什么

發布時間:2021-04-20 16:20:58 來源:億速云 閱讀:292 作者:Leah 欄目:web開發

使用vue實現數據劫持的原理是什么?相信很多沒有經驗的人對此束手無策,為此本文總結了問題出現的原因和解決方法,通過這篇文章希望你能解決這個問題。

Vue的優點

Vue具體輕量級框架、簡單易學、雙向數據綁定、組件化、數據和結構的分離、虛擬DOM、運行速度快等優勢,Vue中頁面使用的是局部刷新,不用每次跳轉頁面都要請求所有數據和dom,可以大大提升訪問速度和用戶體驗。

1. 先定義一個對象

let obj = {
 name: 'jw'
}

2. 定義一個監聽函數

/**
* 判斷監聽的是否是對象
* 如果是對象,就遍歷,并且對每個屬性進行定義get 和 set
*/

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

3.定義一個函數,處理每個屬性

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

ok. 到這里初版已經實現了。嘗試一下吧

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

控制臺輸出:
//數據更新了

以上已經實現設置obj的屬性的時候,被監聽到,并且可以去執行一些代碼了。但是,如果對象里面嵌入了對象呢?比如:

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

執行以下代碼

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

控制臺輸出: 空

4.對監控的obj進行迭代處理

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

再執行以下代碼:

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

控制臺輸出:
//數據更新了

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

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

控制臺輸出:
//空

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

5. 重寫數組的方法

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

再執行以下代碼:

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

控制臺輸出:
//數據更新了

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

總結: Object.defineProperty只是解決了狀態變更后,如何觸發通知的問題,那要通知誰呢?誰會關心那些屬性發生了變化呢?以后再說。

以下完整代碼

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

// vue 數據劫持 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('數據更新了')
   value = val;
  }
 })
}
observer(obj);


// obj.age.old = '50'


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

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

看完上述內容,你們掌握使用vue實現數據劫持的原理是什么的方法了嗎?如果還想學到更多技能或想了解更多相關內容,歡迎關注億速云行業資訊頻道,感謝各位的閱讀!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

vue
AI

阳西县| 永泰县| 永修县| 兴国县| 常山县| 玛曲县| 黔江区| 耿马| 耒阳市| 泾川县| 河东区| 五指山市| 绥棱县| 若尔盖县| 赤峰市| 盖州市| 桐梓县| 临沧市| 长武县| 迭部县| 察雅县| 瓮安县| 临颍县| 西峡县| 民县| 灌云县| 尖扎县| 柘城县| 洞头县| 镇坪县| 辰溪县| 涞源县| 华坪县| 崇礼县| 陆川县| 张北县| 宁陵县| 吴忠市| 吉木萨尔县| 合水县| 仲巴县|