您好,登錄后才能下訂單哦!
使用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實現數據劫持的原理是什么的方法了嗎?如果還想學到更多技能或想了解更多相關內容,歡迎關注億速云行業資訊頻道,感謝各位的閱讀!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。