vue中數組變更不被監測問題

前兩天去玩了,接下來還有挺多學習目標的。今天來寫的以前遇到的問題好了。vue

在以前的項目中,有時候須要對一個數組的某個元素進行改動(如:list[2] = 'b'),卻發現改動了在視圖上確沒有變化,這是爲何呢?這個問題要如何解決呢?git

問題產生的緣由

讓咱們來看看vue中數據劫持的實現。代碼來自:https://github.com/DMQ/mvvmgithub

var data = {name: 'kindeng'};
observe(data);
data.name = 'dmq'; // 哈哈哈,監聽到值變化了 kindeng --> dmq

function observe(data) {
    if (!data || typeof data !== 'object') {
        return;
    }
    // 取出全部屬性遍歷
    Object.keys(data).forEach(function(key) {
	    defineReactive(data, key, data[key]);
	});
};

function defineReactive(data, key, val) {
    observe(val); // 監聽子屬性
    Object.defineProperty(data, key, {
        enumerable: true, // 可枚舉
        configurable: false, 
        get: function() {
            return val;
        },
        set: function(newVal) {
            observe(newVal) //對象新賦值時給新的屬性添加訂閱器
            val = newVal;
        }
    });
}

複製代碼

一、在上述代碼中,只有類型爲‘object’的變量,它下面的屬性纔會繼續遍歷,if (!data || typeof data !== 'object') {return;}。若是是個數組,則不會去遍歷數組裏面的東西。如:數組

var list = [a, b, c]
複製代碼

只對list作了劫持,而list下子屬性則沒有。mvvm

二、list是個引用類型,Object.defineProperty中的set只有在list的地址改變時纔會被觸發,list[2] = 'c'這樣的操做不會引發地址改變。學習

解決問題的方法

vue中提供了set,手動的去添加監聽。ui

//Vue.set( target, key, value )
this.$set(this.list, 2, 'c')
複製代碼

另外,vue重寫了數組的push、pop等方法,使這些操做下的變更能夠被監聽到。this

相關文章
相關標籤/搜索