淺談Vue數(shù)據(jù)響應(yīng)思路之?dāng)?shù)組
之前梳理Vue數(shù)據(jù)響應(yīng)思路 時(shí)沒(méi)有考慮數(shù)組的情況。
js 中數(shù)組有很多實(shí)例方法,其中有一部分會(huì)改變數(shù)組本身的值,比如 push pop shift unshift 等,這些方法被稱為變異方法,這些變異方法也是 Vue 開(kāi)發(fā)中常用的數(shù)組操作方法。那么要實(shí)現(xiàn)對(duì)數(shù)組的觀測(cè),首先要考慮的就是如何截獲這些變異方法的調(diào)用。
簡(jiǎn)單來(lái)說(shuō),Vue 是通過(guò)保持這些數(shù)組變異方法原有功能不變的前提下,對(duì)其功能進(jìn)行擴(kuò)展來(lái)實(shí)現(xiàn)攔截的。具體怎么操作,可以先看一下例子:
function add10(num) {
return num + 10
}
console.log(add10(5)) // 15
const originalAdd10 = add10
add10 = function(num) {
console.log('截獲了add10操作')
return originalAdd10(num)
}
console.log(add10(5)) // '截獲了add10操作'
// 15
該例中,首先使用變量 originalAdd10 緩存 add10 函數(shù),再重新定義 add10 函數(shù),在重新定義的函數(shù)體里就可以執(zhí)行額外增加的功能,比如上例中的 console.log('截獲了add10操作'),然后執(zhí)行緩存的 add10 函數(shù)即 originalAdd10,并將結(jié)果返回,原理大抵如此。
那么,具體可實(shí)現(xiàn)如下:
const mutationMethods = [
'push',
'pop',
'shift',
'unshift',
'splice',
'sort',
'reverse'
]
const arrayMethods = Object.create(Array.prototype)
const arrayProto = Array.prototype
mutationMethods.forEach(method => {
arrayMethods[method] = function (...args) {
const result = arrayProto[method].apply(this, args)
console.log(`我截獲了對(duì)數(shù)組的${method}操作`)
return result
}
})
const arr = ['kobe', 'jordan']
arr.__proto__ = arrayMethods
arr.push('harden') // '我截獲了對(duì)數(shù)組的push操作'
console.log(JSON.stringify(arr)) // '["kobe","jordan","harden"]'
以上,mutationMethods 是所有要攔截的數(shù)組變異方法的集合。
整體思路就是通過(guò)設(shè)置數(shù)組對(duì)象的 __proto__ 屬性的值為一個(gè)新對(duì)象 arrayMethods,以代理數(shù)組 mutationMethods 中的變異方法,并將 arrayMethods 的原型設(shè)置為數(shù)組構(gòu)造函數(shù)本來(lái)的原型,這樣方能保證除卻代理的方法以外,不影響數(shù)組本身的其它方法和屬性。
其中:
const arrayMethods = Object.create(Array.prototype)
以上實(shí)現(xiàn)了 arrayMethods 的原型是數(shù)組構(gòu)造函數(shù)本來(lái)的原型,即 arrayMethods.__proto__ === Array.prototype。
緊接著:
const arrayProto = Array.prototype
這句使用 arrayProto 變量緩存了 Array.prototype。
再然后:
mutationMethods.forEach(method => {
arrayMethods[method] = function (...args) {
const result = arrayProto[method].apply(this, args)
console.log(`我截獲了對(duì)數(shù)組的${method}操作`)
return result
}
})
將 mutationMethods 進(jìn)行循環(huán),在 arrayMethods 對(duì)象上以 mutationMethods 中各元素為 key,即方法名,定義作為攔截器的同名變異方法。
具體:
const result = arrayProto[method].apply(this, args)
執(zhí)行緩存的 Array.prototype,即 arrayProto 中對(duì)應(yīng)的變異方法,并傳入 this 以及 args,也就是將來(lái)調(diào)用該方法的數(shù)組對(duì)象,和調(diào)用該方法時(shí)傳入的參數(shù)(或參數(shù)列表)轉(zhuǎn)化成的參數(shù)數(shù)組,并將結(jié)果給到變量 result。
這里使用了解構(gòu)賦值的方式將參數(shù)(或參數(shù)列表)轉(zhuǎn)化成了參數(shù)數(shù)組,這么做是因?yàn)椴荒艽_定參數(shù)的個(gè)數(shù),所以只能使用 apply(不能用 call),并傳入?yún)?shù)數(shù)組。
之后:
console.log(`我截獲了對(duì)數(shù)組的${method}操作`)
也就是攔截之后要額外執(zhí)行的操作了。
最后:
return result
將數(shù)組原變異方法執(zhí)行的結(jié)果返回,保證原有功能不受影響。
forEach 執(zhí)行完之后:
const arr = ['kobe', 'jordan'] arr.__proto__ = arrayMethods
聲明并初始化 arr,并將 arr 的 __proto__ 指向 arrayMethods,這樣便代理了 mutationMethods 中的變異方法。
最終:
arr.push('harden') // '我截獲了對(duì)數(shù)組的push操作'
console.log(JSON.stringify(arr)) // '["kobe","jordan","harden"]'
數(shù)組對(duì)象手動(dòng)擴(kuò)展的功能以及原功能均正常,實(shí)現(xiàn)了數(shù)組變異方法的攔截。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
vue3.0 CLI - 2.6 - 組件的復(fù)用入門教程
這篇文章主要介紹了 vue3.0 CLI - 2.6 - 組件的復(fù)用,非常不錯(cuò),具有一定的參考借鑒價(jià)值 ,需要的朋友可以參考下2018-09-09
關(guān)于Element-UI可編輯表格的實(shí)現(xiàn)過(guò)程
這篇文章主要介紹了關(guān)于Element-UI可編輯表格的實(shí)現(xiàn)過(guò)程,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-07-07
Vue3.0結(jié)合bootstrap創(chuàng)建多頁(yè)面應(yīng)用
這篇文章主要介紹了Vue3.0結(jié)合bootstrap創(chuàng)建多頁(yè)面應(yīng)用,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2019-05-05
Vue項(xiàng)目在IE瀏覽器頁(yè)面白屏且報(bào)錯(cuò)SCRIPT1010:缺少標(biāo)識(shí)符問(wèn)題
Vue項(xiàng)目在谷歌瀏覽器中正常運(yùn)行,但在IE瀏覽器中出現(xiàn)問(wèn)題,如白屏和控制臺(tái)報(bào)錯(cuò),解決過(guò)程包括檢查IE設(shè)置、調(diào)整編輯器配置、引入兼容性插件、使用productionSourceMap定位錯(cuò)誤、檢查插件依賴和版本,以及重新構(gòu)建項(xiàng)目2024-09-09
單頁(yè)面Vue頁(yè)面刷新出現(xiàn)閃爍問(wèn)題及解決
這篇文章主要介紹了單頁(yè)面Vue頁(yè)面刷新出現(xiàn)閃爍問(wèn)題及解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-07-07
Vue.js實(shí)現(xiàn)的購(gòu)物車功能詳解
這篇文章主要介紹了Vue.js實(shí)現(xiàn)的購(gòu)物車功能,結(jié)合實(shí)例形式分析了vue.js購(gòu)物車的原理、數(shù)值計(jì)算及頁(yè)面元素屬性動(dòng)態(tài)操作相關(guān)實(shí)現(xiàn)技巧,需要的朋友可以參考下2019-01-01
在vue2.x里面簡(jiǎn)單使用socketio問(wèn)題
這篇文章主要介紹了在vue2.x里面簡(jiǎn)單使用socketio問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-10-10
vue實(shí)現(xiàn)購(gòu)物車的小練習(xí)
這篇文章主要為大家詳細(xì)介紹了vue實(shí)現(xiàn)購(gòu)物車的小練習(xí),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-12-12

