日本综合一区二区|亚洲中文天堂综合|日韩欧美自拍一区|男女精品天堂一区|欧美自拍第6页亚洲成人精品一区|亚洲黄色天堂一区二区成人|超碰91偷拍第一页|日韩av夜夜嗨中文字幕|久久蜜综合视频官网|精美人妻一区二区三区

RELATEED CONSULTING
相關(guān)咨詢
選擇下列產(chǎn)品馬上在線溝通
服務時間:8:30-17:00
你可能遇到了下面的問題
關(guān)閉右側(cè)工具欄

新聞中心

這里有您想知道的互聯(lián)網(wǎng)營銷解決方案
思想實驗:如何在Vue中使localStorage具有響應式?

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

成都創(chuàng)新互聯(lián)專注于耀州網(wǎng)站建設服務及定制,我們擁有豐富的企業(yè)做網(wǎng)站經(jīng)驗。 熱誠為您提供耀州營銷型網(wǎng)站建設,耀州網(wǎng)站制作、耀州網(wǎng)頁設計、耀州網(wǎng)站官網(wǎng)定制、小程序開發(fā)服務,打造耀州網(wǎng)絡公司原創(chuàng)品牌,更為您提供耀州網(wǎng)站排名全網(wǎng)營銷落地服務。

讓我們回答這個問題,在解決這個問題時,讓Vue響應式與 localStorage 一起使用。

如果運行以下代碼,則會看到計數(shù)器顯示為靜態(tài)值,并且不會像我們期望的那樣發(fā)生變化,這是因為setInterval在 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: `
  12.     
    Counter: {{ counter }}
  13.     
    Counter is {{ even ? 'even' : 'odd' }}
  14.   
`
  • });
  •  
     
     
     
    1. // some-other-file.js
    2. setInterval(() => {
    3.   const counter = localStorage.getItem("counter");
    4.   localStorage.setItem("counter", +counter + 1);
    5. }, 1000);

    盡管Vue.js實例中的 counter 屬性是響應式的,但它不會因為我們更改了它在 localStorage 中的來源而更改。

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

    Vue 中的響應式

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

     
     
     
     
    1. // core/instance/state.js
    2. function initData () {
    3.   // ...
    4.   observe(data)
    5. }
     
     
     
     
    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. export function defineReactive (obj, key, ...) {
    21.   const dep = new Dep()
    22.   // ...
    23.   Object.defineProperty(obj, key, {
    24.     // ...
    25.     get() {
    26.       // ...
    27.       dep.depend()
    28.       // ...
    29.     },
    30.     set(newVal) {
    31.       // ...
    32.       dep.notify()
    33.     }
    34.   })
    35. }

    所以,為什么 localStorage 不響應?因為它不是具有屬性的對象。

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

    我們可以對 localStorage 做類似的事情嗎?

    覆蓋localStorage函數(shù)

    首先嘗試通過覆蓋localStorage方法來修復最初的示例,以跟蹤哪些組件實例請求了localStorage項目。

     
     
     
     
    1. // LocalStorage項目鍵與依賴它的Vue實例列表之間的映射。
    2. const storeItemSubscribers = {};
    3. const getItem = window.localStorage.getItem;
    4. localStorage.getItem = (key, target) => {
    5.   console.info("Getting", key);
    6.   // 收集依賴的Vue實例
    7.   if (!storeItemSubscribers[key]) storeItemSubscribers[key] = [];
    8.   if (target) storeItemSubscribers[key].push(target);
    9.   // 調(diào)用原始函數(shù) 
    10.   return getItem.call(localStorage, key);
    11. };
    12. const setItem = window.localStorage.setItem;
    13. localStorage.setItem = (key, value) => {
    14.   console.info("Setting", key, value);
    15.   // 更新相關(guān)Vue實例中的值
    16.   if (storeItemSubscribers[key]) {
    17.     storeItemSubscribers[key].forEach((dep) => {
    18.       if (dep.hasOwnProperty(key)) dep[key] = value;
    19.     });
    20.   }
    21.   // 調(diào)用原始函數(shù) 
    22.   setItem.call(localStorage, key, value);
    23. };
     
     
     
     
    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: `
    14.     
      Counter: {{ counter }}
    15.     
      Counter is {{ even ? 'even' : 'odd' }}
    16.   
    `
  • });
  •  
     
     
     
    1. setInterval(() => {
    2.   const counter = localStorage.getItem("counter");
    3.   localStorage.setItem("counter", +counter + 1);
    4. }, 1000);

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

    為了使上面的代碼工作,我們必須向 getItem 傳遞一個對組件實例的引用,這就改變了它的函數(shù)簽名。我們也不能再使用箭頭函數(shù)了,因為否則我們就不會有正確的 this 值。

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

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

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

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

    但是,他們又是誰?

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

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

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

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

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

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

    我們要做的最后一點是使這些生命周期掛鉤成為我們所有組件的一部分,我們可以通過整個項目的全局混合來做到這一點。

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

    現(xiàn)在,當我們運行第一個示例時,我們將獲得一個計數(shù)器,該計數(shù)器每秒增加一個數(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: `
    12.     
      Counter: {{ counter }}
    13.     
      Counter is {{ even ? 'even' : 'odd' }}
    14.   
    `
  • });
  •  
     
     
     
    1. setInterval(() => {
    2.   const counter = localStorage.getItem("counter");
    3.   localStorage.setItem("counter", +counter + 1);
    4. }, 1000);

    我們的思想實驗結(jié)束

    當我們解決了最初的問題時,請記住這主要是一個思想實驗。它缺少一些功能,例如處理已刪除的項目和未安裝的組件實例。它還具有一些限制,例如組件實例的屬性名稱需要與存儲在 localStorage 中的項目相同的名稱。就是說,主要目標是更好地了解Vue響應式在幕后的工作方式并充分利用這一點,因此,我希望你能從所有這些事情中受益。


    分享標題:思想實驗:如何在Vue中使localStorage具有響應式?
    轉(zhuǎn)載來源:http://www.dlmjj.cn/article/dhjcgsg.html