自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

思想實(shí)驗(yàn):如何在Vue中使localStorage具有響應(yīng)式?

開(kāi)發(fā) 前端
響應(yīng)式是Vue.js的最大特色之一。如果你不知道幕后情況,它也是最神秘的地方之一。例如,為什么它不能用于對(duì)象和數(shù)組,而不能用于諸如 localStorage 之類的其他東西?

響應(yīng)式是Vue.js的最大特色之一。如果你不知道幕后情況,它也是最神秘的地方之一。例如,為什么它不能用于對(duì)象和數(shù)組,而不能用于諸如 localStorage 之類的其他東西?

[[333547]]

讓我們回答這個(gè)問(wèn)題,在解決這個(gè)問(wèn)題時(shí),讓Vue響應(yīng)式與 localStorage 一起使用。

如果運(yùn)行以下代碼,則會(huì)看到計(jì)數(shù)器顯示為靜態(tài)值,并且不會(huì)像我們期望的那樣發(fā)生變化,這是因?yàn)閟etInterval在 localStorage 中更改了該值。

  1. new Vue({ 
  2.   el: "#counter", 
  3.   data: () => ({ 
  4.     counter: localStorage.getItem("counter") 
  5.   }), 
  6.   computed: { 
  7.     even() { 
  8.       return this.counter % 2 == 0; 
  9.     } 
  10.   }, 
  11.   template: `<div> 
  12.     <div>Counter: {{ counter }}</div> 
  13.     <div>Counter is {{ even ? 'even' : 'odd' }}</div> 
  14.   </div>
  15. }); 
  1. // some-other-file.js 
  2. setInterval(() => { 
  3.   const counter = localStorage.getItem("counter"); 
  4.   localStorage.setItem("counter", +counter + 1); 
  5. }, 1000); 

盡管Vue.js實(shí)例中的 counter 屬性是響應(yīng)式的,但它不會(huì)因?yàn)槲覀兏牧怂?localStorage 中的來(lái)源而更改。

有多種解決方案,最好的也許是使用Vuex,并保持存儲(chǔ)值與 localStorage 同步。但如果我們需要像本例中那樣簡(jiǎn)單的東西呢?我們要深入了解一下Vue.js的響應(yīng)式系統(tǒng)是如何工作的。

Vue 中的響應(yīng)式

當(dāng)Vue初始化組件實(shí)例時(shí),它將觀察data選項(xiàng)。這意味著它將遍歷數(shù)據(jù)中的所有屬性,并使用 Object.defineProperty 將它們轉(zhuǎn)換為getter/setter。通過(guò)為每個(gè)屬性設(shè)置自定義設(shè)置器,Vue可以知道屬性何時(shí)發(fā)生更改,并且可以通知需要對(duì)更改做出反應(yīng)的依賴者。它如何知道哪些依賴者依賴于一個(gè)屬性?通過(guò)接入getters,它可以在計(jì)算的屬性、觀察者函數(shù)或渲染函數(shù)訪問(wèn)數(shù)據(jù)屬性時(shí)進(jìn)行注冊(cè)。

  1. // core/instance/state.js 
  2. function initData () { 
  3.   // ... 
  4.   observe(data) 
  1. // core/observer/index.js 
  2. export function observe (value) { 
  3.   // ... 
  4.   new Observer(value) 
  5.   // ... 
  6.  
  7. export class Observer { 
  8.   // ... 
  9.   constructor (value) { 
  10.     // ... 
  11.     this.walk(value) 
  12.   } 
  13.    
  14.   walk (obj) { 
  15.     const keys = Object.keys(obj) 
  16.     for (let i = 0; i < keys.length; i++) { 
  17.       defineReactive(obj, keys[i]) 
  18.     } 
  19.   } 
  20. }  
  21.  
  22. export function defineReactive (obj, key, ...) { 
  23.   const dep = new Dep() 
  24.   // ... 
  25.   Object.defineProperty(obj, key, { 
  26.     // ... 
  27.     get() { 
  28.       // ... 
  29.       dep.depend() 
  30.       // ... 
  31.     }, 
  32.     set(newVal) { 
  33.       // ... 
  34.       dep.notify() 
  35.     } 
  36.   }) 

所以,為什么 localStorage 不響應(yīng)?因?yàn)樗皇蔷哂袑傩缘膶?duì)象。

但是等一下,我們也不能用數(shù)組定義getter和setter,但Vue中的數(shù)組仍然是反應(yīng)式的。這是因?yàn)閿?shù)組在Vue中是一種特殊情況。為了擁有響應(yīng)式的數(shù)組,Vue在后臺(tái)重寫(xiě)了數(shù)組方法,并與Vue的響應(yīng)式系統(tǒng)進(jìn)行了修補(bǔ)。

我們可以對(duì) localStorage 做類似的事情嗎?

覆蓋localStorage函數(shù)

首先嘗試通過(guò)覆蓋localStorage方法來(lái)修復(fù)最初的示例,以跟蹤哪些組件實(shí)例請(qǐng)求了localStorage項(xiàng)目。

  1. // LocalStorage項(xiàng)目鍵與依賴它的Vue實(shí)例列表之間的映射。 
  2. const storeItemSubscribers = {}; 
  3.  
  4. const getItem = window.localStorage.getItem; 
  5. localStorage.getItem = (key, target) => { 
  6.   console.info("Getting", key); 
  7.  
  8.   // 收集依賴的Vue實(shí)例 
  9.   if (!storeItemSubscribers[key]) storeItemSubscribers[key] = []; 
  10.   if (target) storeItemSubscribers[key].push(target); 
  11.  
  12.   // 調(diào)用原始函數(shù)  
  13.   return getItem.call(localStorage, key); 
  14. }; 
  15.  
  16. const setItem = window.localStorage.setItem; 
  17. localStorage.setItem = (key, value) => { 
  18.   console.info("Setting", key, value); 
  19.  
  20.   // 更新相關(guān)Vue實(shí)例中的值 
  21.   if (storeItemSubscribers[key]) { 
  22.     storeItemSubscribers[key].forEach((dep) => { 
  23.       if (dep.hasOwnProperty(key)) dep[key] = value; 
  24.     }); 
  25.   } 
  26.  
  27.   // 調(diào)用原始函數(shù)  
  28.   setItem.call(localStorage, key, value); 
  29. }; 
  1. new Vue({ 
  2.   el: "#counter", 
  3.   data: function() { 
  4.     return { 
  5.       counter: localStorage.getItem("counter", this) // 我們現(xiàn)在需要傳遞“this” 
  6.     } 
  7.   }, 
  8.   computed: { 
  9.     even() { 
  10.       return this.counter % 2 == 0; 
  11.     } 
  12.   }, 
  13.   template: `<div> 
  14.     <div>Counter: {{ counter }}</div> 
  15.     <div>Counter is {{ even ? 'even' : 'odd' }}</div> 
  16.   </div>
  17. }); 
  1. setInterval(() => { 
  2.   const counter = localStorage.getItem("counter"); 
  3.   localStorage.setItem("counter", +counter + 1); 
  4. }, 1000); 

在這個(gè)例子中,我們重新定義了 getItem 和 setItem,以便收集和通知依賴 localStorage 項(xiàng)目的組件。在新的 getItem 中,我們注意到哪個(gè)組件請(qǐng)求了哪個(gè)項(xiàng)目,在 setItems 中,我們聯(lián)系所有請(qǐng)求該項(xiàng)目的組件,并重寫(xiě)它們的數(shù)據(jù)屬性。

為了使上面的代碼工作,我們必須向 getItem 傳遞一個(gè)對(duì)組件實(shí)例的引用,這就改變了它的函數(shù)簽名。我們也不能再使用箭頭函數(shù)了,因?yàn)榉駝t我們就不會(huì)有正確的 this 值。

如果我們想做得更好,就必須更深入地挖掘。例如,我們?nèi)绾卧诓伙@式傳遞依賴者的情況下跟蹤它們?

思想實(shí)驗(yàn):如何在Vue中使localStorage具有響應(yīng)式?

Vue如何收集依賴關(guān)系

為了獲得啟發(fā),我們可以回到Vue的響應(yīng)式系統(tǒng)。我們之前曾看到,訪問(wèn)數(shù)據(jù)屬性時(shí),數(shù)據(jù)屬性的 getter 將使調(diào)用者訂閱該屬性的進(jìn)一步更改。但是它怎么知道是誰(shuí)做的調(diào)用呢?當(dāng)我們得到一個(gè)數(shù)據(jù)屬性時(shí),它的 getter 函數(shù)沒(méi)有任何關(guān)于調(diào)用者是誰(shuí)的輸入。Getter函數(shù)沒(méi)有輸入,它怎么知道誰(shuí)要注冊(cè)為依賴者呢?

每個(gè)數(shù)據(jù)屬性維護(hù)一個(gè)需要在Dep類中進(jìn)行響應(yīng)的依賴項(xiàng)列表。如果我們?cè)诖祟愔羞M(jìn)行更深入的研究,可以看到只要在注冊(cè)依賴項(xiàng)時(shí)就已經(jīng)在靜態(tài)目標(biāo)變量中定義了依賴項(xiàng)。這個(gè)目標(biāo)是由一個(gè)非常神秘的Watche類確定的。實(shí)際上,當(dāng)數(shù)據(jù)屬性更改時(shí),將實(shí)際通知這些觀察程序,并且它們將啟動(dòng)組件的重新渲染或計(jì)算屬性的重新計(jì)算。

但是,他們又是誰(shuí)?

當(dāng)Vue使 data 選項(xiàng)可觀察時(shí),它還會(huì)為每個(gè)計(jì)算出的屬性函數(shù)以及所有watch函數(shù)(不應(yīng)與Watcher類混為一談)以及每個(gè)組件實(shí)例的render函數(shù)創(chuàng)建watcher。觀察者就像這些函數(shù)的伴侶。他們主要做兩件事:

  • 當(dāng)它們被創(chuàng)建時(shí),它們會(huì)評(píng)估函數(shù)。這將觸發(fā)依賴關(guān)系的集合。
  • 當(dāng)他們被通知他們所依賴的一個(gè)值發(fā)生變化時(shí),他們會(huì)重新運(yùn)行他們的函數(shù)。這將最終重新計(jì)算一個(gè)計(jì)算出的屬性或重新渲染整個(gè)組件。

在觀察者調(diào)用其負(fù)責(zé)的函數(shù)之前,有一個(gè)重要的步驟發(fā)生了:他們將自己設(shè)置為Dep類中靜態(tài)變量的目標(biāo)。這樣可以確保在訪問(wèn)響應(yīng)式數(shù)據(jù)屬性時(shí)將它們注冊(cè)為從屬。

追蹤誰(shuí)調(diào)用了localStorage

我們無(wú)法完全做到這一點(diǎn),因?yàn)槲覀儫o(wú)法使用Vue的內(nèi)部機(jī)制。但是,我們可以使用Vue的想法,即觀察者可以在調(diào)用其負(fù)責(zé)的函數(shù)之前,將目標(biāo)設(shè)置為靜態(tài)屬性。我們能否在調(diào)用 localStorage 之前設(shè)置對(duì)組件實(shí)例的引用?

如果我們假設(shè)在設(shè)置 data 選項(xiàng)時(shí)調(diào)用了 localStorage,則可以將其插入 beforeCreate 和 created 中。這兩個(gè)掛鉤在初始化data選項(xiàng)之前和之后都會(huì)被觸發(fā),因此我們可以設(shè)置一個(gè)目標(biāo)變量,然后清除該變量,并引用當(dāng)前組件實(shí)例(我們可以在生命周期掛鉤中訪問(wèn)該實(shí)例)。然后,在我們的自定義獲取器中,我們可以將該目標(biāo)注冊(cè)為依賴項(xiàng)。

我們要做的最后一點(diǎn)是使這些生命周期掛鉤成為我們所有組件的一部分,我們可以通過(guò)整個(gè)項(xiàng)目的全局混合來(lái)做到這一點(diǎn)。

  1. // LocalStorage項(xiàng)目鍵與依賴它的Vue實(shí)例列表之間的映射 
  2. const storeItemSubscribers = {}; 
  3.  
  4. // 當(dāng)前正在初始化的Vue實(shí)例 
  5. let target = undefined
  6.  
  7. const getItem = window.localStorage.getItem; 
  8. localStorage.getItem = (key) => { 
  9.   console.info("Getting", key); 
  10.  
  11.   // 收集依賴的Vue實(shí)例 
  12.   if (!storeItemSubscribers[key]) storeItemSubscribers[key] = []; 
  13.   if (target) storeItemSubscribers[key].push(target); 
  14.  
  15.   // 調(diào)用原始函數(shù)  
  16.   return getItem.call(localStorage, key); 
  17. }; 
  18.  
  19. const setItem = window.localStorage.setItem; 
  20. localStorage.setItem = (key, value) => { 
  21.   console.info("Setting", key, value); 
  22.  
  23.   // 更新相關(guān)Vue實(shí)例中的值 
  24.   if (storeItemSubscribers[key]) { 
  25.     storeItemSubscribers[key].forEach((dep) => { 
  26.       if (dep.hasOwnProperty(key)) dep[key] = value; 
  27.     }); 
  28.   } 
  29.    
  30.   // 調(diào)用原始函數(shù)  
  31.   setItem.call(localStorage, key, value); 
  32. }; 
  33.  
  34. Vue.mixin({ 
  35.   beforeCreate() { 
  36.     console.log("beforeCreate", this._uid); 
  37.     target = this
  38.   }, 
  39.   created() { 
  40.     console.log("created", this._uid); 
  41.     target = undefined
  42.   } 
  43. }); 

現(xiàn)在,當(dāng)我們運(yùn)行第一個(gè)示例時(shí),我們將獲得一個(gè)計(jì)數(shù)器,該計(jì)數(shù)器每秒增加一個(gè)數(shù)字。

  1. new Vue({ 
  2.   el: "#counter", 
  3.   data: () => ({ 
  4.     counter: localStorage.getItem("counter") 
  5.   }), 
  6.   computed: { 
  7.     even() { 
  8.       return this.counter % 2 == 0; 
  9.     } 
  10.   }, 
  11.   template: `<div class="component"> 
  12.     <div>Counter: {{ counter }}</div> 
  13.     <div>Counter is {{ even ? 'even' : 'odd' }}</div> 
  14.   </div>
  15. }); 
  1. setInterval(() => { 
  2.   const counter = localStorage.getItem("counter"); 
  3.   localStorage.setItem("counter", +counter + 1); 
  4. }, 1000); 

我們的思想實(shí)驗(yàn)結(jié)束

當(dāng)我們解決了最初的問(wèn)題時(shí),請(qǐng)記住這主要是一個(gè)思想實(shí)驗(yàn)。它缺少一些功能,例如處理已刪除的項(xiàng)目和未安裝的組件實(shí)例。它還具有一些限制,例如組件實(shí)例的屬性名稱需要與存儲(chǔ)在 localStorage 中的項(xiàng)目相同的名稱。就是說(shuō),主要目標(biāo)是更好地了解Vue響應(yīng)式在幕后的工作方式并充分利用這一點(diǎn),因此,我希望你能從所有這些事情中受益。

 

責(zé)任編輯:趙寧寧 來(lái)源: 今日頭條
相關(guān)推薦

2024-07-08 08:43:19

2020-02-10 10:23:03

VueJSX前端

2020-06-09 11:35:30

Vue 3響應(yīng)式前端

2019-07-01 13:34:22

vue系統(tǒng)數(shù)據(jù)

2021-01-22 11:47:27

Vue.js響應(yīng)式代碼

2023-12-06 07:43:56

Vue如何定義事件

2021-12-02 05:50:35

Vue3 插件Vue應(yīng)用

2021-05-19 14:25:19

前端開(kāi)發(fā)技術(shù)

2017-08-30 17:10:43

前端JavascriptVue.js

2011-08-10 09:31:41

Hibernateunion

2021-03-09 07:27:40

Kafka開(kāi)源分布式

2015-08-27 09:46:09

swiftAFNetworkin

2021-06-09 09:36:18

DjangoElasticSearLinux

2022-05-17 08:25:10

TypeScript接口前端

2022-06-23 08:00:53

PythonDateTime模塊

2024-01-18 08:37:33

socketasyncio線程

2019-09-16 19:00:48

Linux變量

2014-07-02 09:47:06

SwiftCocoaPods

2020-04-09 10:18:51

Bash循環(huán)Linux

2024-09-06 11:34:15

RustAI語(yǔ)言
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)