Vue響應(yīng)式原理

前言の前言:寫得極其不通順戈钢,只是快速做個(gè)零碎總結(jié),后續(xù)會(huì)不斷潤色

vue2與vue3在實(shí)現(xiàn)上的區(qū)別

vue2實(shí)現(xiàn)數(shù)據(jù)劫持使用的是Object.defineProperty, Vue3使用的是Proxy

共同需要解決的問題

  1. 響應(yīng)式的對(duì)象
  2. 響應(yīng)式的數(shù)組

看看vue響應(yīng)式的表現(xiàn)

  • 定義在data中的數(shù)據(jù)才是響應(yīng)式的
  • 【對(duì)象】使用vm.a =1視圖不會(huì)更新择份,需要用Vue.set 或 vm.$set
  • 給對(duì)象批量添加屬性O(shè)bject.assign(this.obj, { a:1, b:2})不行田度;得用this.obj=Object.assign({}, this.obj, {a:1, b:2})象颖,即加上原對(duì)象一起混入
  • 【數(shù)組】arr[1] = 2不行布疙,得用Vue.set或者vm.$set
  • arr.length=2改長度不行,需要用splice
  • 動(dòng)態(tài)在vm.data中加入屬性是無效的辆床,必須在初始化前聲明
  • Vue更新dom是異步的佳晶;只要偵聽到數(shù)據(jù)變化,Vue 將開啟一個(gè)隊(duì)列讼载,并緩沖在同一事件循環(huán)中發(fā)生的所有數(shù)據(jù)變更

在了解這些vue2的表現(xiàn)后轿秧,我們往原理層面看看中跌,到底為什么會(huì)是這樣

使用Object.defineProperty

表面上,這個(gè)Object的API Object.defineProperty 和直接使用字面量給對(duì)象賦值的效果貌似是一樣的淤刃,那么實(shí)際上他們到底有什么區(qū)別呢
(1)定義方式

// 使用Object.defineProperty給對(duì)象定義屬性
let obj = {};
// 參數(shù): 對(duì)象晒他、屬性名/Symbol、描述符(里面的叫做鍵逸贾、值)
Object.defineProperty( obj, "name", {
    configurable: false,   // 1.是否能刪除 2.描述符是否能修改
    enumerable: false, // 是否可枚舉
    value: undefined, // 值
    writable: false, // 能否被賦值運(yùn)算符=賦值
    get: undefined,  // 訪問該屬性時(shí)調(diào)用陨仅,傳入this,返回結(jié)果用作屬性值
    set: undefined  // 修改屬性值時(shí)調(diào)用铝侵,傳入修改值和this
} )

// 使用字面量的方式
obj.name = undefined;

很顯然灼伤,Object.defineProperty可以使用描述來控制該屬性的配置、枚舉咪鲜、修改狐赡,攔截get和set的過程,而這正是我們實(shí)現(xiàn)數(shù)據(jù)劫持所需要的特性疟丙。

此外颖侄,如果想要批量添加屬性的話,可以使用Object.defineProperties,示例

const obj = {};
Object.defineProperties(obj, {
    name: {
        value: "123",
        writable: true
    },
    name2: {
        value: "456"
    }
});

基本的響應(yīng)式實(shí)現(xiàn)

// 數(shù)據(jù)
const data = {
    name: "123",
    name2: "456"
};

// 將data變成響應(yīng)式
observer(data);

function observer(target) {
    // 非數(shù)組享郊、對(duì)象 直接返回
    if(typeof target !== 'object' || typeof target !== null) {
        return target;
    }

    // 數(shù)組览祖、對(duì)象實(shí)現(xiàn)數(shù)據(jù)劫持
    for(let key in target) {
        defineReactive(target, key, target[key]);
    }
}

// 使用defineProperty實(shí)現(xiàn)簡(jiǎn)單的數(shù)據(jù)劫持
function defineReactive(target, key, value) {
    Object.defineProperty(target, key, {
        value,
        get: function() {
            return value;
        },
        set: function(newVal) {
            value = newVal;
            // todo: 更新視圖
        }
    })
}

這里補(bǔ)一張圖:


在這里插入圖片描述

vue實(shí)際上是通過發(fā)布訂閱者模式進(jìn)行數(shù)據(jù)驅(qū)動(dòng)視圖更新的。
在我理解炊琉,發(fā)布訂閱者模式可以高度概括為一句話:控制并發(fā)布數(shù)據(jù)的Subject會(huì)在數(shù)據(jù)發(fā)生變動(dòng)時(shí)通知所有注冊(cè)&訂閱了的Observer進(jìn)行更新展蒂。

細(xì)化.復(fù)雜對(duì)象

const data = {
    name: "123",
    name2: "456",
    name3: {
        firstName: "c",
        lastName: "xk"
    }
};

// 將data變成響應(yīng)式的數(shù)據(jù)
observer(data);

// observer的實(shí)現(xiàn):遞歸偵聽
function observer(target) {
    if(typeof target !== 'object' || typeof target !== null) {
        return target;
    }

    // s數(shù)組、對(duì)象使用defineProperty實(shí)現(xiàn)簡(jiǎn)單的數(shù)據(jù)劫持
    for(let key in target) {
        observer(newVal);
        Object.defineProperty(target, key, {
            value,
            get: function() {
                return value;
            },
            set: function(newVal) {
                observer(newVal);
                value = newVal;
                // todo: 更新視圖
            }
        })
    }
}

當(dāng)數(shù)據(jù)是多層級(jí)對(duì)象時(shí)苔咪,之前的實(shí)現(xiàn)就無法偵聽到深層次屬性的變化了锰悼,所以需要使用遞歸進(jìn)行優(yōu)化;另外团赏,如果set的時(shí)候箕般,給定的也是個(gè)對(duì)象 如 obj.name = { name }, 那么也無法監(jiān)聽,所以這里也需要遞歸優(yōu)化下舔清,遞歸時(shí)間復(fù)雜度很高隘世,所以在vue2在遇到復(fù)雜對(duì)象時(shí)性能不會(huì)很好,vue3使用proxy解決了這個(gè)問題鸠踪。

細(xì)化.數(shù)組

const {
  arguments
} = require("file-loader");
const {
  method
} = require("lodash");

const data = {
  name: "123",
  name2: "456",
  name3: {
    firstName: "c",
    lastName: "xk"
  },
  names: ['1', '2', '3'];
};

// 利用Array原型創(chuàng)建新原型
const oldArrayProto = Array.prototype;
const newArrayProto = Object.create(oldArrayProto);
['push', 'pop', 'shift', 'unshift', 'splice'].forEach(methodName => {
  newArrayProto[methodName] = function () {
    // todo: 更新視圖
    oldArrayProto[methodName].call(this, ...arguments);
  }
});

// 將data變成響應(yīng)式的數(shù)據(jù)
observer(data);

// observer的實(shí)現(xiàn):遞歸偵聽
function observer(target) {
  if (typeof target !== 'object' || typeof target !== null) {
    return target;
  }

  // 將_proto_指向新原型
  if (Array.isArray(target)) {
    target._proto_ = newArrayProto;
  }

  // s數(shù)組、對(duì)象使用defineProperty實(shí)現(xiàn)簡(jiǎn)單的數(shù)據(jù)劫持
  for (let key in target) {
    observer(newVal);
    Object.defineProperty(target, key, {
      value,
      get: function () {
        return value;
      },
      set: function (newVal) {
        observer(newVal);
        value = newVal;
        // todo: 更新視圖
      }
    })
  }
}

Object.create方法的作用是Creates an object that has the specified prototype or that has null prototype复斥,即創(chuàng)建有原型指向舊對(duì)象的新對(duì)象营密。

后續(xù)將繼續(xù)補(bǔ)充完善

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市目锭,隨后出現(xiàn)的幾起案子评汰,更是在濱河造成了極大的恐慌纷捞,老刑警劉巖,帶你破解...
    沈念sama閱讀 218,858評(píng)論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件被去,死亡現(xiàn)場(chǎng)離奇詭異主儡,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)惨缆,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,372評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門糜值,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人坯墨,你說我怎么就攤上這事寂汇。” “怎么了捣染?”我有些...
    開封第一講書人閱讀 165,282評(píng)論 0 356
  • 文/不壞的土叔 我叫張陵骄瓣,是天一觀的道長。 經(jīng)常有香客問我耍攘,道長榕栏,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,842評(píng)論 1 295
  • 正文 為了忘掉前任蕾各,我火速辦了婚禮扒磁,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘示损。我一直安慰自己渗磅,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,857評(píng)論 6 392
  • 文/花漫 我一把揭開白布检访。 她就那樣靜靜地躺著始鱼,像睡著了一般。 火紅的嫁衣襯著肌膚如雪脆贵。 梳的紋絲不亂的頭發(fā)上医清,一...
    開封第一講書人閱讀 51,679評(píng)論 1 305
  • 那天,我揣著相機(jī)與錄音卖氨,去河邊找鬼会烙。 笑死,一個(gè)胖子當(dāng)著我的面吹牛筒捺,可吹牛的內(nèi)容都是我干的柏腻。 我是一名探鬼主播,決...
    沈念sama閱讀 40,406評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼系吭,長吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼五嫂!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,311評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤沃缘,失蹤者是張志新(化名)和其女友劉穎躯枢,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體槐臀,經(jīng)...
    沈念sama閱讀 45,767評(píng)論 1 315
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡锄蹂,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,945評(píng)論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了水慨。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片得糜。...
    茶點(diǎn)故事閱讀 40,090評(píng)論 1 350
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖讥巡,靈堂內(nèi)的尸體忽然破棺而出掀亩,到底是詐尸還是另有隱情,我是刑警寧澤欢顷,帶...
    沈念sama閱讀 35,785評(píng)論 5 346
  • 正文 年R本政府宣布槽棍,位于F島的核電站,受9級(jí)特大地震影響抬驴,放射性物質(zhì)發(fā)生泄漏炼七。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,420評(píng)論 3 331
  • 文/蒙蒙 一布持、第九天 我趴在偏房一處隱蔽的房頂上張望豌拙。 院中可真熱鬧,春花似錦题暖、人聲如沸按傅。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,988評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽唯绍。三九已至,卻和暖如春枝誊,著一層夾襖步出監(jiān)牢的瞬間况芒,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,101評(píng)論 1 271
  • 我被黑心中介騙來泰國打工叶撒, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留绝骚,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,298評(píng)論 3 372
  • 正文 我出身青樓祠够,卻偏偏與公主長得像压汪,于是被迫代替她去往敵國和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子古瓤,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,033評(píng)論 2 355

推薦閱讀更多精彩內(nèi)容