vue實現(xiàn)雙向綁定原理

原理

vue數(shù)據(jù)雙向綁定通過‘數(shù)據(jù)劫持’ + 訂閱發(fā)布模式實現(xiàn)

數(shù)據(jù)劫持

指的是在訪問或者修改對象的某個屬性時壳鹤,通過一段代碼攔截這個行為,進行額外的操作或者修改返回結(jié)果

典型的有
1.Object.defineProperty()
2.es6中Proxy對象

vue2.x使用Object.defineProperty();
vue3.x使用Proxy;

訂閱發(fā)布模式

定義:對象間的一種一對多的依賴關(guān)系丑念,當一個對象的狀態(tài)發(fā)生改變時爱只,所有依賴于它的對象都將得到通知
訂閱發(fā)布模式中事件統(tǒng)一由處理中心處理,訂閱者發(fā)布者互不干擾凰兑。
優(yōu)點:實現(xiàn)更多的控制妥粟,做權(quán)限處理,節(jié)流控制之類吏够,例如:發(fā)布了很多消息勾给,但是不是所有訂閱者都要接收

// 實現(xiàn)一個處理中心
let event = {
  clientList: {}, // 訂閱事件列表
  // 訂閱
  on(key, fn){
    // 如果這個事件沒有被訂閱,那么創(chuàng)建一個列表用來存放事件
    if(!this.clientList[key]) {
      this.clientList[key] = []
    }
    // 將事件放入已有的事件列表中
    this.clientList[key].push(fn);
  },
  // 發(fā)布
  trigger(type, args){
    let fns = this.clientList[type] // 拿到這個事件的所有監(jiān)聽
    if(!fns || fns.length === 0){  // 如果沒有這條消息的訂閱者
      return false
    }
    // 如果存在這個事件的訂閱锅知,那么遍歷事件列表播急,觸發(fā)對應監(jiān)聽
    fns.forEach(fn => {
      // 可以在此處添加過濾等處理
      fn(args)
    })
  }
}

vue中如何實現(xiàn)

利用Object.defineProperty();把內(nèi)部解耦為三部分
Observer: 遞歸的監(jiān)聽對象上的所有屬性,當屬性改變時觸發(fā)對應的watcher
watcher(觀察者):當蔣婷的數(shù)據(jù)值修改時售睹,執(zhí)行相應的回調(diào)函數(shù)桩警,更新模板內(nèi)容
dep:鏈接observer和watcher,每一個observer對應一個dep,內(nèi)部維護一個數(shù)組昌妹,保存與該observer相關(guān)的watcher

proxy實現(xiàn)觀察者模式

觀察者模式(Observer mode)指的是函數(shù)自動觀察數(shù)據(jù)對象捶枢,一旦對象有變化,函數(shù)就會自動執(zhí)行

const person = observable({
  name: '張三',
  age: 20
});

function print() {
  console.log(`${person.name}, ${person.age}`)
}

observe(print);
person.name = '李四';
// 輸出
// 李四, 20

代碼中飞崖。對象person是觀察目標烂叔,函數(shù)print是觀察者。一旦數(shù)據(jù)發(fā)生變化固歪,print就會自動執(zhí)行

使用proxy實現(xiàn)一個最簡單觀察者模式蒜鸡,即實現(xiàn)observable和observe這兩個函數(shù)。
思路是observable函數(shù)返回一個原始對象的proxy代理牢裳,攔截復制操作逢防。觸發(fā)充當觀察者的各個函數(shù)

const queue = new Set();

const observe = fn => queue.add(fn);
const observable = obj => new Proxy(obj, {set});

function set(target, key, value, receiver) {
  const result = Reflect.set(target, key, value, receiver);
  queue.forEach(observer => observer());
  return result;
} 

上面代碼中,先定義了一個Set集合蒲讯,所有觀察者函數(shù)都放進這個集合胞四,然后,observable函數(shù)返回原始對象的代理伶椿,攔截賦值操作。
攔截函數(shù)set中氓侧,自動執(zhí)行所有觀察者

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末脊另,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子约巷,更是在濱河造成了極大的恐慌偎痛,老刑警劉巖,帶你破解...
    沈念sama閱讀 221,198評論 6 514
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件独郎,死亡現(xiàn)場離奇詭異踩麦,居然都是意外死亡枚赡,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,334評論 3 398
  • 文/潘曉璐 我一進店門谓谦,熙熙樓的掌柜王于貴愁眉苦臉地迎上來贫橙,“玉大人,你說我怎么就攤上這事反粥÷啵” “怎么了?”我有些...
    開封第一講書人閱讀 167,643評論 0 360
  • 文/不壞的土叔 我叫張陵才顿,是天一觀的道長莫湘。 經(jīng)常有香客問我,道長郑气,這世上最難降的妖魔是什么幅垮? 我笑而不...
    開封第一講書人閱讀 59,495評論 1 296
  • 正文 為了忘掉前任,我火速辦了婚禮尾组,結(jié)果婚禮上忙芒,老公的妹妹穿的比我還像新娘。我一直安慰自己演怎,他們只是感情好匕争,可當我...
    茶點故事閱讀 68,502評論 6 397
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著爷耀,像睡著了一般甘桑。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上跑杭,一...
    開封第一講書人閱讀 52,156評論 1 308
  • 那天德谅,我揣著相機與錄音萨螺,去河邊找鬼。 笑死慰技,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的掏颊。 我是一名探鬼主播,決...
    沈念sama閱讀 40,743評論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼盆偿,長吁一口氣:“原來是場噩夢啊……” “哼准浴!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起兄裂,我...
    開封第一講書人閱讀 39,659評論 0 276
  • 序言:老撾萬榮一對情侶失蹤晰奖,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后啃匿,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 46,200評論 1 319
  • 正文 獨居荒郊野嶺守林人離奇死亡溯乒,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 38,282評論 3 340
  • 正文 我和宋清朗相戀三年裆悄,在試婚紗的時候發(fā)現(xiàn)自己被綠了光稼。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 40,424評論 1 352
  • 序言:一個原本活蹦亂跳的男人離奇死亡艾君,死狀恐怖冰垄,靈堂內(nèi)的尸體忽然破棺而出权她,到底是詐尸還是另有隱情,我是刑警寧澤蝴罪,帶...
    沈念sama閱讀 36,107評論 5 349
  • 正文 年R本政府宣布拾徙,位于F島的核電站感局,受9級特大地震影響暂衡,放射性物質(zhì)發(fā)生泄漏狂巢。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,789評論 3 333
  • 文/蒙蒙 一唧领、第九天 我趴在偏房一處隱蔽的房頂上張望斩个。 院中可真熱鬧驯杜,春花似錦、人聲如沸滚局。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,264評論 0 23
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽糯景。三九已至,卻和暖如春丑孩,著一層夾襖步出監(jiān)牢的瞬間灭贷,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,390評論 1 271
  • 我被黑心中介騙來泰國打工仗岖, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留览妖,地道東北人。 一個月前我還...
    沈念sama閱讀 48,798評論 3 376
  • 正文 我出身青樓檩电,卻偏偏與公主長得像,于是被迫代替她去往敵國和親俐末。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 45,435評論 2 359

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