<span id="mktg5"></span>

<i id="mktg5"><meter id="mktg5"></meter></i>

        <label id="mktg5"><meter id="mktg5"></meter></label>
        最新文章專題視頻專題問答1問答10問答100問答1000問答2000關鍵字專題1關鍵字專題50關鍵字專題500關鍵字專題1500TAG最新視頻文章推薦1 推薦3 推薦5 推薦7 推薦9 推薦11 推薦13 推薦15 推薦17 推薦19 推薦21 推薦23 推薦25 推薦27 推薦29 推薦31 推薦33 推薦35 推薦37視頻文章20視頻文章30視頻文章40視頻文章50視頻文章60 視頻文章70視頻文章80視頻文章90視頻文章100視頻文章120視頻文章140 視頻2關鍵字專題關鍵字專題tag2tag3文章專題文章專題2文章索引1文章索引2文章索引3文章索引4文章索引5123456789101112131415文章專題3
        問答文章1 問答文章501 問答文章1001 問答文章1501 問答文章2001 問答文章2501 問答文章3001 問答文章3501 問答文章4001 問答文章4501 問答文章5001 問答文章5501 問答文章6001 問答文章6501 問答文章7001 問答文章7501 問答文章8001 問答文章8501 問答文章9001 問答文章9501
        當前位置: 首頁 - 科技 - 知識百科 - 正文

        Vue如何實現響應式系統

        來源:懂視網 責編:小采 時間:2020-11-27 22:11:40
        文檔

        Vue如何實現響應式系統

        Vue如何實現響應式系統:前言 最近深入學習了Vue實現響應式的部分源碼,將我的些許收獲和思考記錄下來,希望能對看到這篇文章的人有所幫助。有什么問題歡迎指出,大家共同進步。 什么是響應式系統 一句話概括:數據變更驅動視圖更新。這樣我們就可以以數據驅動的思維來編寫我們
        推薦度:
        導讀Vue如何實現響應式系統:前言 最近深入學習了Vue實現響應式的部分源碼,將我的些許收獲和思考記錄下來,希望能對看到這篇文章的人有所幫助。有什么問題歡迎指出,大家共同進步。 什么是響應式系統 一句話概括:數據變更驅動視圖更新。這樣我們就可以以數據驅動的思維來編寫我們

        前言

        最近深入學習了Vue實現響應式的部分源碼,將我的些許收獲和思考記錄下來,希望能對看到這篇文章的人有所幫助。有什么問題歡迎指出,大家共同進步。

        什么是響應式系統

        一句話概括:數據變更驅動視圖更新。這樣我們就可以以“數據驅動”的思維來編寫我們的代碼,更多的關注業務,而不是dom操作。其實Vue響應式的實現是一個變化追蹤和變化應用的過程。

        vue響應式原理

        以數據劫持方式,攔截數據變化;以依賴收集方式,觸發視圖更新。利用es5 Object.defineProperty攔截數據的setter、getter;getter收集依賴,setter觸發依賴更新,而組件render也會變為一個watcher callback被加入相應數據的依賴中。

        發布訂閱

        利用發布訂閱設計模式實現,Observer作為發布者,Watcher作為訂閱者,兩者無直接交互,通過Dep進行統一調度。
        Observer負責攔截get, set;get時觸發dep添加依賴,set時調度dep發布;添加Watcher時會觸發訂閱數據的get,并加入到dep調度中心的訂閱者隊列中。

        以下的UML類圖是Vue實現響應式功能的類,以及他們之間的引用關系。

        只包含部分屬性方法

        上圖中的類已經標識的蠻清楚了,但是還是需要一個調用關系圖,讓調用過程更加清晰,如下圖所示。

        響應式data對象中,每一項key的劫持get/set函數都閉包了Dep調度實例,這張圖顯示了一個key更改過程中的數據流轉。

        部分源碼

        數據變更過程中的訂閱/發布模型上圖已經清晰的展示了,從圖中我們已經知道了可以通過增加watcher來訂閱某一項數據的變更。那么,我們只需要把組件render作為一個watcher訂閱的話,數據驅動視圖的渲染豈不是水到渠成了。Vue正是這么做的!
        以下代碼片段來自Vue.prototype._mount函數

        callHook(vm, 'beforeMount')
        vm._watcher = new Watcher(vm, () => {
         vm._update(vm._render(), hydrating)
        }, noop)
        hydrating = false
        // manually mounted instance, call mounted on self
        // mounted is called for render-created child components in its inserted hook
        if (vm.$vnode == null) {
         vm._isMounted = true
         callHook(vm, 'mounted')
        }

        一些問題思考

        #person賦值新的對象,新對象里的屬性是否也是響應式的呢?

        var vm = new Vue({
         el: '#app',
         data: () => ({
         person: null
         })
        })
        vm.person = {name: 'zs'}
        setTimeout(() => {
         // 更改name
         vm.person.name = 'finally zs'
        }, 3000)

        答案:是響應式的。

        原因:因為Vue劫持set時,會對value再次做observe,源碼如下。

        function reactiveSetter (newVal) {
         /* ...省略部分代碼 */
         // 這里會再次對新的value做攔截
         childOb = observe(newVal)
         dep.notify()
        }

        #當我們監聽多層屬性時,上層引用變更,是否會觸發回調?

        var vm = new Vue({
         data: () => ({
         person: {name: '令狐洋蔥'}
         }),
         watch: {
         'person.name'(val) {
         console.log('name updated', val)
         }
         }
        })
        vm.person = {}

        答案:會。

        原因:person.name作為一個表達式傳入Watcher時,會被解析成類似這樣的函數

        () => {this.vm.person.name}

        這樣就會先觸發person get, 然后觸發name get;所以我們配置的回調函數,不僅僅加入到了name依賴中,person也有。

        #接著上個問題,person如果被賦值了新的對象,老對象和老對象上的依賴如何垃圾回收的?

      1. 老對象的回收:由于老對象的直接引用只有vue實例上的person,person切換到了新的引用,所以老對象沒有引用了,就會被回收掉。
      2. 老對象上的依賴dep,watcher的依賴里還存在;但是在run執行時,會調用watcher的get() 獲取當前值;get中會執行新的依賴收集,并在收集完畢后,清空老的依賴。
      3. 具體源碼如下:

        /**
        * Evaluate the getter, and re-collect dependencies.
        */
        get () {
         pushTarget(this)
         const value = this.getter.call(this.vm, this.vm)
         // "touch" every property so they are all tracked as
         // dependencies for deep watching
         if (this.deep) {
         traverse(value)
         }
         popTarget()
         this.cleanupDeps()
         return value
        }

        #當我們多次同步修改name時,回調函數是否會觸發多次?

        var vm = new Vue({
         data: () => ({
         person: {name: '令狐洋蔥'}
         }),
         watch: {
         'person.name': (val) {
         console.log('name updated: ' + val)
         }
         }
        })
        vm.person = {name: 'zs'}
        vm.person.name = '無敵'

        答案: 不會,因為watch回調函數執行是異步的,且會去重。可以通過sync強制配置成同步run,就會執行2次了。

        自己實現一個響應式系統

        只包含核心功能,具體源碼可以看這里https://github.com/Zenser/z-vue,歡迎來star。

        實現功能非常基礎啦,重在理解,功能不全的。

        Observer

        class Observe {
         constructor(obj) {
         Object.keys(obj).forEach(prop => {
         reactive(obj, prop, obj[prop])
         })
         }
        }
        function reactive(obj, prop) {
         let value = obj[prop]
         // 閉包綁定依賴
         let dep = new Dep()
         Object.defineProperty(obj, prop, {
         configurable: true,
         enumerable: true,
         get() {
         //利用js單線程,在get時綁定訂閱者
         if (Dep.target) {
         // 綁定訂閱者
         dep.addSub(Dep.target)
         }
         return value
         },
         set(newVal) {
         value = newVal
         // 更新時,觸發訂閱者更新
         dep.notify()
         }
         })
         // 對象監聽
         if (typeof value === 'object' && value !== null) {
         Object.keys(value).forEach(valueProp => {
         reactive(value, valueProp)
         })
         } 
        }

        Dep

        class Dep {
         constructor() {
         this.subs = []
         }
         addSub(sub) {
         if (this.subs.indexOf(sub) === -1) {
         this.subs.push(sub)
         }
         }
         notify() {
         this.subs.forEach(sub => {
         const oldVal = sub.value
         sub.cb && sub.cb(sub.get(), oldVal)
         })
         }
        }

        Watcher

        class Watcher {
         constructor(data, exp, cb) {
         this.data = data
         this.exp = exp
         this.cb = cb
         this.get()
         }
         get() {
         Dep.target = this
         this.value = (function calcValue(data, prop) {
         for (let i = 0, len = prop.length; i < len; i++ ) {
         data = data[prop[i]]
         }
         return data
         })(this.data, this.exp.split('.'))
         Dep.target = null
         return this.value
         }
        }

        參考文檔:https://cn.vuejs.org/v2/guide/reactivity.html

        聲明:本網頁內容旨在傳播知識,若有侵權等問題請及時與本網聯系,我們將在第一時間刪除處理。TEL:177 7030 7066 E-MAIL:11247931@qq.com

        文檔

        Vue如何實現響應式系統

        Vue如何實現響應式系統:前言 最近深入學習了Vue實現響應式的部分源碼,將我的些許收獲和思考記錄下來,希望能對看到這篇文章的人有所幫助。有什么問題歡迎指出,大家共同進步。 什么是響應式系統 一句話概括:數據變更驅動視圖更新。這樣我們就可以以數據驅動的思維來編寫我們
        推薦度:
        • 熱門焦點

        最新推薦

        猜你喜歡

        熱門推薦

        專題
        Top
        主站蜘蛛池模板: 2022久久国产精品免费热麻豆| 精品日韩亚洲AV无码| 亚洲高清在线mv| 波霸在线精品视频免费观看| 日韩免费观看一级毛片看看| 亚洲国产日韩女人aaaaaa毛片在线| 日韩精品无码免费专区午夜不卡| 亚洲国产成人a精品不卡在线| 亚洲日韩一区二区一无码| 亚洲人成电影网站免费| 亚洲伊人精品综合在合线| 真人做人试看60分钟免费视频| 亚洲成人黄色网址| 91黑丝国产线观看免费| 亚洲一区二区三区免费观看| 思思re热免费精品视频66| 亚洲国产夜色在线观看| 大地资源二在线观看免费高清 | 久久99九九国产免费看小说| 久久亚洲AV成人无码电影| 在线美女免费观看网站h| 麻豆亚洲av熟女国产一区二| 亚洲成年人免费网站| 亚洲三级中文字幕| 免费被黄网站在观看| 免费国产a理论片| AV在线亚洲男人的天堂| 最近中文字幕大全免费版在线| 亚洲高清视频在线观看| 性xxxxx免费视频播放| 亚洲男同gay片| 国产亚洲精品拍拍拍拍拍| 日韩免费的视频在线观看香蕉| 亚洲日韩乱码久久久久久| 99视频在线精品免费观看6| 免费国产在线精品一区| 亚洲AV中文无码乱人伦下载 | 91在线老王精品免费播放| 亚洲国产韩国一区二区| 国产精品二区三区免费播放心| 猫咪免费人成网站在线观看入口|