日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術(shù)文章
文章詳情頁

淺析vue偵測數(shù)據(jù)的變化之基本實現(xiàn)

瀏覽:127日期:2022-09-28 18:29:54
目錄一、Object的變化偵測二、關(guān)于 Object 的問題三、Array 的變化偵測3.1、背景3.2、實現(xiàn)四、關(guān)于 Array 的問題一、Object的變化偵測

下面我們就來模擬偵測數(shù)據(jù)變化的邏輯。

強(qiáng)調(diào)一下我們要做的事情:數(shù)據(jù)變化,通知到外界(外界再做一些自己的邏輯處理,比如重新渲染視圖)。

開始編碼之前,我們首先得回答以下幾個問題:

1.如何偵測對象的變化?

使用 Object.defineProperty()。讀數(shù)據(jù)的時候會觸發(fā) getter,修改數(shù)據(jù)會觸發(fā) setter。 只有能偵測對象的變化,才能在數(shù)據(jù)發(fā)生變化的時候發(fā)出通知

2.當(dāng)數(shù)據(jù)發(fā)生變化的時候,我們通知誰?

通知用到數(shù)據(jù)的地方。而數(shù)據(jù)可以用在模板中,也可以用在 vm.$watch() 中,地方不同,行為也不相同,比如這里要渲染模板,那里要進(jìn)行其他邏輯。所以干脆抽象出一個類。當(dāng)數(shù)據(jù)變化的時候通知它,再由它去通知其他地方。 這個類起名叫 Watcher。就是一個中介。

3.依賴誰?

通知誰,就依賴誰,依賴 Watcher。

4.何時通知?

修改數(shù)據(jù)的時候。也就是 setter 中通知

5.何時收集依賴?

因為要通知用數(shù)據(jù)的地方。用數(shù)據(jù)就得讀數(shù)據(jù),我們就可以在讀數(shù)據(jù)的時候收集,也就是在 getter 中收集

6.收集到哪里?

可以在每個屬性里面定義一個數(shù)組,與該屬性有關(guān)的依賴都放里面

編碼如下(可直接運(yùn)行):

// 全局變量,用于存儲依賴let globalData = undefined;// 將數(shù)據(jù)轉(zhuǎn)為響應(yīng)式function defineReactive (obj,key,val) { // 依賴列表 let dependList = [] Object.defineProperty(obj, key, { enumerable: true, configurable: true, get: function () {// 收集依賴(Watcher)globalData && dependList.push(globalData)return val }, set: function reactiveSetter (newVal) {if(val === newVal){ return}// 通知依賴項(Watcher)dependList.forEach(w => { w.update(newVal, val)})val = newVal } });}// 依賴class Watcher{ constructor(data, key, callback){this.data = data;this.key = key;this.callback = callback;this.val = this.get(); } // 這段代碼可以將自己添加到依賴列表中 get(){// 將依賴保存在 globalDataglobalData = this;// 讀數(shù)據(jù)的時候收集依賴let value = this.data[this.key]globalData = undefinedreturn value; } // 數(shù)據(jù)改變時收到通知,然后再通知到外界 update(newVal, oldVal){this.callback(newVal, oldVal) }}/* 以下是測試代碼 */let data = {};// 將 name 屬性轉(zhuǎn)為響應(yīng)式defineReactive(data, ’age’, ’88’)// 當(dāng)數(shù)據(jù) age 改變時,會通知到 Watcher,再由 Watcher 通知到外界new Watcher(data, ’age’, (newVal, oldVal) => { console.log(`外界:newVal = ${newVal} ; oldVal = ${oldVal}`)})data.age -= 1 // 控制臺輸出: 外界:newVal = 87 ; oldVal = 88

在控制臺下繼續(xù)執(zhí)行 data.age -= 1,則會輸出 外界:newVal = 86 ; oldVal = 87。

附上一張 Data、defineReactive、dependList、Watcher和外界的關(guān)系圖。

淺析vue偵測數(shù)據(jù)的變化之基本實現(xiàn)

首先通過 defineReactive() 方法將 data 轉(zhuǎn)為響應(yīng)式(defineReactive(data, ’age’, ’88’))。

外界通過 Watcher 讀取數(shù)據(jù)(let value = this.data[this.key]),數(shù)據(jù)的 getter 則會被觸發(fā),于是通過 globalData 收集Watcher。

當(dāng)數(shù)據(jù)被修改(data.age -= 1), 會觸發(fā) setter,會通知依賴(dependList),依賴則會通知 Watcher(w.update(newVal, val)),最后 Watcher 再通知給外界。

二、關(guān)于 Object 的問題

思考一下:上面的例子,繼續(xù)執(zhí)行 delete data.age 會通知到外界嗎?

不會。因為不會觸發(fā) setter。請接著看:

<!DOCTYPE html><html lang='en'><head> <meta charset='UTF-8'> <meta name='viewport' content='width=device-width, initial-scale=1.0'> <title>Document</title> <script src='https://cdn.jsdelivr.net/npm/vue/dist/vue.js'></script></head><body> <div id=’app’><section> {{ p1.name }} {{ p1.age }}</section> </div><script>const app = new Vue({ el: ’#app’, data: {p1: { name: ’ph’, age: 18} }})</script></body></html>

運(yùn)行后,頁面會顯示 ph 18。我們知道更改數(shù)據(jù),視圖會重新渲染,于是在控制臺執(zhí)行 delete app.p1.name,發(fā)現(xiàn)頁面沒有變化。這與上面示例中執(zhí)行 delete data.age 一樣,都不會觸發(fā)setter,也就不會通知到外界。

為了解決這個問題,Vue提供了兩個 API(稍后將介紹它們):vm.$set 和 vm.$delete。

如果你繼續(xù)執(zhí)行 app.$delete(app.p1, ’age’),你會發(fā)現(xiàn)頁面沒有任何信息了(name 屬性已經(jīng)用 delete 刪除了,只是當(dāng)時沒有重新渲染而已)。

注:如果這里執(zhí)行 app.p1.sex = ’man’,用到數(shù)據(jù) p1 的地方也不會被通知到,這個問題可以通過 vm.$set 解決。

三、Array 的變化偵測3.1、背景

假如數(shù)據(jù)是 let data = {a:1, b:[11, 22]},通過 Object.defineProperty 將其轉(zhuǎn)為響應(yīng)式之后,我們修改數(shù)據(jù) data.a = 2,會通知到外界,這個好理解;同理 data.b = [11, 22, 33] 也會通知到外界,但如果換一種方式修改數(shù)據(jù) b,就像這樣 data.b.push(33),是不會通知到外界的,因為沒走 setter。請看示例:

function defineReactive(obj, key, val) { Object.defineProperty(obj, key, { enumerable: true, configurable: true, get: function () {console.log(`get val = ${val}`)return val }, set: function reactiveSetter (newVal) {if(val === newVal){ return}console.log(`set val = ${newVal}; oldVal = ${val}`)val = newVal } });}// 以下是測試代碼 {1}let data = {}defineReactive(data, ’a’, [11,22])data.a.push(33) // get val = 11,22 (沒有觸發(fā) setter) {2} data.a // get val = 11,22,33 data.a = 1 // set val = 1; oldVal = 11,22,33(觸發(fā) setter)

通過 push() 方法改變數(shù)組的值,確實沒有觸發(fā) setter(行{2}),也就不能通知外界。這里好像說明了一個問題:通過 Object.definePropery() 方法,只能將對象轉(zhuǎn)為響應(yīng)式,不能將數(shù)組轉(zhuǎn)為響應(yīng)式。

其實 Object.definePropery() 可以將數(shù)組轉(zhuǎn)為響應(yīng)式。請看示例:

// 繼續(xù)上面的例子,將測試代碼(行{1})改為:let data = []defineReactive(data, ’0’, 11)data[0] = 22 // set val = 22; oldVal = 11data.push(33) // 不會觸發(fā) {10}

雖然 Object.definePropery() 可以將數(shù)組轉(zhuǎn)為響應(yīng)式,但通過 data.push(33)(行{10})這種方式修改數(shù)組,仍然不會通知到外界。

所以在 Vue 中,將數(shù)據(jù)轉(zhuǎn)為響應(yīng)式,用了兩套方式:對象使用 Object.defineProperty();數(shù)組則使用另一套。

3.2、實現(xiàn)

es6 中可以用 Proxy 偵測數(shù)組的變化。請看示例:

let data = [11,22]let p = new Proxy(data, { set: function(target, prop, value, receiver) {target[prop] = value;console.log(’property set: ’ + prop + ’ = ’ + value);return true; } })console.log(p)p.push(33)/*輸出:[ 11, 22 ]property set: 2 = 33property set: length = 3*/

es6 以前就稍微麻煩點(diǎn),可以使用攔截器。原理是:當(dāng)我們執(zhí)行 [].push() 時會調(diào)用數(shù)組原型(Array.prototype)中的方法。我們在 [].push() 和 Array.prototype 之間增加一個攔截器,以后調(diào)用 [].push() 時先執(zhí)行攔截器中的 push() 方法,攔截器中的 push() 在調(diào)用 Array.prototype 中的 push() 方法。請看示例:

// 數(shù)組原型let arrayPrototype = Array.prototype// 創(chuàng)建攔截器let interceptor = Object.create(arrayPrototype)// 將攔截器與原始數(shù)組的方法關(guān)聯(lián)起來;(’push,pop,unshift,shift,splice,sort,reverse’).split(’,’).forEach(method => { let origin = arrayPrototype[method]; Object.defineProperty(interceptor, method, {value: function(...args){ console.log(`攔截器: args = ${args}`) return origin.apply(this, args);},enumerable: false,writable: true,configurable: true })});// 測試let arr1 = [’a’]let arr2 = [10]arr1.push(’b’)// 偵測數(shù)組 arr2 的變化Object.setPrototypeOf(arr2, interceptor) // {20}arr2.push(11) // 攔截器: args = 11arr2.unshift(22) // 攔截器: args = 22

這個例子將能改變數(shù)組自身內(nèi)容的 7 個方法都加入到了攔截器。如果需要偵測哪個數(shù)組的變化,就將該數(shù)組的原型指向攔截器(行{20})。當(dāng)我們通過 push 等 7 個方法修改該數(shù)組時,則會在攔截器中觸發(fā),從而可以通知外界。

到這里,我們只完成了偵測數(shù)組變化的任務(wù)。

數(shù)據(jù)變化,通知到外界。上文編碼的實現(xiàn)只是針對 Object 數(shù)據(jù),而這里需要針對 Array 數(shù)據(jù)。

我們也來思考一下同樣的問題:

1.如何偵測數(shù)組的變化?

攔截器

2.當(dāng)數(shù)據(jù)發(fā)生變化的時候,我們通知誰?

Watcher

3.依賴誰?

Watcher

4.何時通知?

修改數(shù)據(jù)的時候。攔截器中通知。

5.何時收集依賴?

因為要通知用數(shù)據(jù)的地方。用數(shù)據(jù)就得讀數(shù)據(jù)。在讀數(shù)據(jù)的時候收集。這和對象收集依賴是一樣的。 {a: [11,22]} 比如我們要使用 a 數(shù)組,肯定得訪問對象的屬性 a。

6.收集到哪里?

對象是在每個屬性中收集依賴,但這里得考慮數(shù)組在攔截器中能觸發(fā)依賴,位置可能得調(diào)整

就到這里,不在繼續(xù)展開了。接下來的文章中,我會將 vue 中與數(shù)據(jù)偵測相關(guān)的源碼摘出來,配合本文,簡單分析一下。

四、關(guān)于 Array 的問題

// 需要自己引入 vue.js。后續(xù)也盡可能只羅列核心代碼<div id=’app’><section> {{ p1[0] }} {{ p1[1] }}</section></div><script>const app = new Vue({ el: ’#app’, data: {p1: [’ph’, ’18’] }})</script>

運(yùn)行后在頁面顯示 ph 18,控制臺執(zhí)行 app.p1[0] = ’lj’ 頁面沒反應(yīng),因為數(shù)組只有調(diào)用指定的 7 個方法才能通過攔截器通知外界。如果執(zhí)行 app.$set(app.p1, 0, ’pm’) 頁面內(nèi)容會變成 pm 18。

以上就是淺析vue偵測數(shù)據(jù)的變化之基本實現(xiàn)的詳細(xì)內(nèi)容,更多關(guān)于vue偵測數(shù)據(jù)的變化的資料請關(guān)注好吧啦網(wǎng)其它相關(guān)文章!

標(biāo)簽: Vue
相關(guān)文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
秋霞影院一区二区三区| 精品女同一区二区三区在线观看| 91精品蜜臀一区二区三区在线| 狠狠久久婷婷| www.九色在线| 国产高清精品二区| 日韩三级一区| 视频在线观看一区| 国产91精品对白在线播放| 精品国产一区二| 国产精品一区二区美女视频免费看 | 久久亚洲国产| av在线最新| 欧美日韩夜夜| 一级欧美视频| 午夜久久福利| 亚洲一区二区毛片| 亚洲91久久| 岛国精品一区| 久久精品超碰| 亚洲精品自拍| 国产调教精品| 国产成人免费av一区二区午夜| 亚洲精品乱码久久久久久蜜桃麻豆| 日韩午夜高潮| 日韩精品三级| 免费一级欧美片在线观看网站 | 国产在线观看www| 欧美羞羞视频| 亚洲主播在线| 亚洲精品一二三**| 国产免费av一区二区三区| 免费在线成人| re久久精品视频| 亚洲色图国产| 精品一区二区三区在线观看视频| 国产一区二区三区四区大秀| 香蕉精品久久| 久久国产亚洲精品| 国产综合精品| 国产精品1区| 欧美va亚洲va日韩∨a综合色| 免费久久99精品国产自在现线| 国产乱码精品一区二区三区四区 | 国产精品福利在线观看播放| 视频一区在线播放| 久久一区精品| 色偷偷偷在线视频播放| 中文字幕一区二区三区日韩精品| 91综合网人人| 日韩av一区二区三区| 精品视频在线观看网站| 欧洲一区二区三区精品| 国产精品av久久久久久麻豆网| 性欧美长视频| 在线手机中文字幕| 欧美精品福利| 美女亚洲一区| 欧美aa一级| 国产精品.xx视频.xxtv| 免费在线观看成人| 成人自拍av| 久久精品一区二区国产| 久久aⅴ国产紧身牛仔裤| 精品国产黄a∨片高清在线| 日韩在线播放一区二区| 中文字幕在线视频久| 国产私拍福利精品视频二区| 在线国产一区| 国产96在线亚洲| 国产女人18毛片水真多18精品| 99国产精品99久久久久久粉嫩| 97精品国产一区二区三区| 久久国产人妖系列| 日本成人在线视频网站| 午夜亚洲福利| 午夜在线播放视频欧美| 久久久久99| 高清不卡亚洲| 日本成人在线一区| 久久99伊人| 亚洲男女自偷自拍| 西西人体一区二区| 免费人成精品欧美精品| 日本国产欧美| 国产精品一区二区免费福利视频| 日韩精品欧美大片| 欧美精品国产| 加勒比视频一区| xxxxx性欧美特大| 久久婷婷久久| 蜜桃视频一区二区| 亚洲精品亚洲人成在线观看| 影音国产精品| 一区二区国产在线| 国产欧美欧美| 国精品产品一区| 婷婷成人在线| 亚洲一本视频| 亚洲天堂日韩在线| 欧美有码在线| 亚洲国产欧美日本视频| 亚洲一区二区三区高清| 91伊人久久| 99久久婷婷| 亚洲乱码视频| 蜜桃视频欧美| 欧美成人精品午夜一区二区| 欧美高清不卡| 麻豆成人在线观看| 中文久久精品| 美女精品一区二区| 国产精品呻吟| 日产午夜精品一线二线三线| 欧美jjzz| 欧美亚洲一区二区三区| 青青久久av| 免费视频一区二区三区在线观看| 狠狠操综合网| 国产精品极品国产中出| 蜜臀久久久久久久| 成人精品视频| 日韩精品91亚洲二区在线观看| 精品中文字幕一区二区三区av| 日韩精品三级| 欧美日韩免费观看一区=区三区| 国产毛片一区二区三区| 国产欧洲在线| 国产精品极品| 亚洲免费影视| 久久久久网站| 国产日韩一区二区三区在线播放| 日韩午夜av在线| 久久精品国产68国产精品亚洲| 九九九精品视频| 欧美日韩亚洲一区在线观看| 在线免费观看亚洲| aa国产精品| 欧美不卡在线| 在线日韩中文| 欧美日韩国产探花| 在线日韩一区| av一区二区高清| 日韩视频不卡| 极品日韩av| 亚洲香蕉网站| 欧美在线亚洲| 亚洲激情偷拍| 中文字幕一区日韩精品| 亚洲精品a级片| 亚洲天堂av资源在线观看| 麻豆亚洲精品| 免费久久99精品国产| 亚洲人成网站在线在线观看| 亚洲永久精品唐人导航网址| 亚洲精品一区二区在线播放∴| 国产一区二区三区自拍| 欧美午夜不卡| 一区二区国产在线观看| 午夜欧美视频| 久久午夜视频| 国产精品啊啊啊| 久久五月天小说| 国产欧美亚洲一区| 夜夜嗨一区二区| 日韩精品视频在线看| 樱桃视频成人在线观看| 日本亚洲最大的色成网站www| 日本久久精品| 亚洲一区二区三区高清| 久久成人福利| 蜜桃视频第一区免费观看| 久久一区精品| 91精品尤物| 中文在线中文资源| 午夜日韩影院| 欧美日韩激情| 给我免费播放日韩视频| 日韩中文字幕| 女主播福利一区| 91精品精品| 精品中文字幕一区二区三区 | 日韩二区三区在线观看| 成人av二区| 91日韩在线| 亚洲最新av| 天堂网av成人| 中文字幕在线视频久| 免费看一区二区三区| 日本a口亚洲| 亚洲精品伊人| 亚洲免费一区三区| 免费观看不卡av| 欧美日韩免费观看视频| 美腿丝袜在线亚洲一区| 国产精品亚洲一区二区在线观看| 日韩三级一区| 啪啪亚洲精品| 天海翼精品一区二区三区| 国产亚洲在线观看|