日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

詳解Vue中的MVVM原理和實現方法

瀏覽:276日期:2023-01-07 17:17:57

下面由我阿巴阿巴的詳細走一遍Vue中MVVM原理的實現,這篇文章大家可以學習到:

1.Vue數據雙向綁定核心代碼模塊以及實現原理

2.訂閱者-發布者模式是如何做到讓數據驅動視圖、視圖驅動數據再驅動視圖

3.如何對元素節點上的指令進行解析并且關聯訂閱者實現視圖更新

一、思路整理

實現的流程圖:

詳解Vue中的MVVM原理和實現方法

我們要實現一個類MVVM簡單版本的Vue框架,就需要實現一下幾點:

1、實現一個數據監聽Observer,對數據對象的所有屬性進行監聽,數據發生變化可以獲取到最新值通知訂閱者。

2、實現一個解析器Compile解析頁面節點指令,初始化視圖。

3、實現一個觀察者Watcher,訂閱數據變化同時綁定相關更新函數。并且將自己放入觀察者集合Dep中。Dep是Observer和Watcher的橋梁,數據改變通知到Dep,然后Dep通知相應的Watcher去更新視圖。

二、實現

以下采用ES6的寫法,比較簡潔,所以大概在300多行代碼實現了一個簡單的MVVM框架。

1、實現html頁面

按Vue的寫法在頁面定義好一些數據跟指令,引入了兩個JS文件。先實例化一個MVue的對象,傳入我們的el,data,methods這些參數。待會再看Mvue.js文件是什么?

html

<body> <div id='app'> <h2>{{person.name}} --- {{person.age}}</h2> <h3>{{person.fav}}</h3> <h3>{{person.a.b}}</h3> <ul> <li>1</li> <li>2</li> <li>3</li> </ul> <h3>{{msg}}</h3> <div v-text='msg'></div> <div v-text='person.fav'></div> <div v-html='htmlStr'></div> <input type='text' v-model='msg'> <button v-on:click='click111'>按鈕on</button> <button @click='click111'>按鈕@</button> </div> <script src='http://m.b3g6.com/bcjs/MVue.js'></script> <script src='http://m.b3g6.com/bcjs/Observer.js'></script> <script> let vm = new MVue({ el: ’#app’, data: { person: { name: ’星哥’, age: 18, fav: ’姑娘’, a: { b: ’787878’ } }, msg: ’學習MVVM實現原理’, htmlStr: ’<h4>大家學的怎么樣</h4>’, }, methods: { click111() { console.log(this) this.person.name = ’學習MVVM’ // this.$data.person.name = ’學習MVVM’ } } }) </script></body>

2、實現解析器和觀察者

MVue.js

// 先創建一個MVue類,它是一個入口Class MVue { construction(options) { this.$el = options.el this.$data = options.data this.$options = options } if(this.$el) { // 1.實現一個數據的觀察者 --先看解析器,再看Obeserver new Observer(this.$data) // 2.實現一個指令解析器 new Compile(this.$el,this) }}​// 定義一個Compile類解析元素節點和指令class Compile { constructor(el,vm) { // 判斷el是否是元素節點對象,不是就通過DOM獲取 this.el = this.isElementNode(el) ? el : document.querySelector(el) this.vm = vm // 1.獲取文檔碎片對象,放入內存中可以減少頁面的回流和重繪 const fragment = this.node2Fragment(this.el) // 2.編輯模板 this.compile(fragment) // 3.追加子元素到根元素(還原頁面) this.el.appendChild(fragment) } // 將元素插入到文檔碎片中 node2Fragment(el) { const f = document.createDocumnetFragment(); let firstChild while(firstChild = el.firstChild) { // appendChild // 將已經存在的節點再次插入,那么原來位置的節點自動刪除,并在新的位置重新插入。 f.appendChild(firstChild) } // 此處執行完,頁面已經沒有元素節點了 return f } // 解析模板 compile(frafment) { // 1.獲取子節點 conts childNodes = fragment.childNodes; [...childNodes].forEach(child => { if(this.isElementNode(child)) {// 是元素節點// 編譯元素節點this.compileElement(child) } else {// 文本節點// 編譯文本節點this.compileText(child) } // 嵌套子節點進行遍歷解析 if(child.childNodes && child.childNodes.length) {this.compule(child) } }) } // 判斷是元素節點還是屬性節點 isElementNode(node) { // nodeType屬性返回 以數字值返回指定節點的節點類型。1-元素節點 2-屬性節點 return node.nodeType === 1 } // 編譯元素節點 compileElement(node) { // 獲得元素屬性集合 const attributes = node.attributes [...attributes].forEach(attr => { const {name, value} = attr if(this.isDirective(name)) { // 判斷屬性是不是以v-開頭的指令// 解析指令(v-mode v-text v-on:click 等...)const [, dirctive] = name.split(’-’)const [dirName, eventName] = dirctive.split(’:’)// 初始化視圖 將數據渲染到視圖上compileUtil[dirName](node, value, this.vm, eventName)// 刪除有指令的標簽上的屬性node.removeAttribute(’v-’ + dirctive) } else if (this.isEventName(name)) { //判斷屬性是不是以@開頭的指令// 解析指令let [, eventName] = name.split(’@’)compileUtil[’on’](node,val,this.vm, eventName)// 刪除有指令的標簽上的屬性node.removeAttribute(’@’ + eventName) } else if(this.isBindName(name)) { //判斷屬性是不是以:開頭的指令// 解析指令let [, attrName] = name.split(’:’)compileUtil[’bind’](node,val,this.vm, attrName)// 刪除有指令的標簽上的屬性node.removeAttribute(’:’ + attrName) } }) } // 編譯文本節點 compileText(node) { const content = node.textContent if(/{{(.+?)}}/.test(content)) { compileUtil[’text’](node, content, this.vm) } } // 判斷屬性是不是指令 isDirective(attrName) { return attrName.startsWith(’v-’) } // 判斷屬性是不是以@開頭的事件指令 isEventName(attrName) { return attrName.startsWith(’@’) } // 判斷屬性是不是以:開頭的事件指令 isBindName(attrName) { return attrName.startsWith(’:’) }}​​// 定義一個對象,針對不同指令執行不同操作const compileUtil = { // 解析參數(包含嵌套參數解析),獲取其對應的值 getVal(expre, vm) { return expre.split(’.’).reduce((data, currentVal) => { return data[currentVal] }, vm.$data) }, // 獲取當前節點內參數對應的值 getgetContentVal(expre,vm) { return expre.replace(/{{(.+?)}}/g, (...arges) => { return this.getVal(arges[1], vm) }) }, // 設置新值 setVal(expre, vm, inputVal) { return expre.split(’.’).reduce((data, currentVal) => { return data[currentVal] = inputVal }, vm.$data) }, // 指令解析:v-test test(node, expre, vm) { let value; if(expre.indexOf(’{{’) !== -1) { // 正則匹配{{}}里的內容 value = expre.replace(/{{(.+?)}}/g, (...arges) => {// new watcher這里相關的先可以不看,等后面講解寫到觀察者再回頭看。這里是綁定觀察者實現 的效果是通過改變數據會觸發視圖,即數據=》視圖。// 沒有new watcher 不影響視圖初始化(頁面參數的替換渲染)。// 訂閱數據變化,綁定更新函數。new watcher(vm, arges[1], () => { // 確保 {{person.name}}----{{person.fav}} 不會因為一個參數變化都被成新值 this.updater.textUpdater(node, this.getgetContentVal(expre,vm))})return this.getVal(arges[1],vm) }) } else { // 同上,先不看 // 數據=》視圖 new watcher(vm, expre, (newVal) => { // 找不到{}說明是test指令,所以當前節點只有一個參數變化,直接用回調函數傳入的新值 this.updater.textUpdater(node, newVal) }) value = this.getVal(expre,vm) } // 將數據替換,更新到視圖上 this.updater.textUpdater(node,value) }, //指令解析: v-html html(node, expre, vm) { const value = this.getVal(expre, vm) // 同上,先不看 // 綁定觀察者 數據=》視圖 new watcher(vm, expre (newVal) => { this.updater.htmlUpdater(node, newVal) }) // 將數據替換,更新到視圖上 this.updater.htmlUpdater(node, newVal) }, // 指令解析:v-mode model(node,expre, vm) { const value = this.getVal(expre, vm) // 同上,先不看 // 綁定觀察者 數據=》視圖 new watcher(vm, expre, (newVal) => { this.updater.modelUpdater(node, newVal) }) // input框 視圖=》數據=》視圖 node.addEventListener(’input’, (e) => { //設置新值 - 將input值賦值到v-model綁定的參數上 this.setVal(expre, vm, e.traget.value) }) // 將數據替換,更新到視圖上 this.updater.modelUpdater(node, value) }, // 指令解析: v-on on(node, expre, vm, eventName) { // 或者指令綁定的事件函數 let fn = vm.$option.methods && vm.$options.methods[expre] // 監聽函數并調用 node.addEventListener(eventName,fn.bind(vm),false) }, // 指令解析: v-bind bind(node, expre, vm, attrName) { const value = this.getVal(expre,vm) this.updater.bindUpdate(node, attrName, value) }// updater對象,管理不同指令對應的更新方法updater: { // v-text指令對應更新方法 textUpdater(node, value) { node.textContent = value }, // v-html指令對應更新方法 htmlUpdater(node, value) { node.innerHTML = value }, // v-model指令對應更新方法 modelUpdater(node,value) { node.value = value }, // v-bind指令對應更新方法 bindUpdate(node, attrName, value) { node[attrName] = value } },}

3、實現數據劫持監聽

我們有了數據監聽,還需要一個觀察者可以觸發更新視圖。因為需要數據改變才能觸發更新,所有還需要一個橋梁Dep收集所有觀察者(觀察者集合),連接Observer和Watcher。數據改變通知Dep,Dep通知相應的觀察者進行視圖更新。

Observer.js

// 定義一個觀察者class watcher { constructor(vm, expre, cb) { this.vm = vm this.expre = expre this.cb =cb // 把舊值保存起來 this.oldVal = this.getOldVal() } // 獲取舊值 getOldVal() { // 將watcher放到targe值中 Dep.target = this // 獲取舊值 const oldVal = compileUtil.getVal(this.expre, this.vm) // 將target值清空 Dep.target = null return oldVal } // 更新函數 update() { const newVal = compileUtil.getVal(this.expre, this.vm) if(newVal !== this.oldVal) { this.cb(newVal) } }}​​// 定義一個觀察者集合class Dep { constructor() { this.subs = [] } // 收集觀察者 addSub(watcher) { this.subs.push(watcher) } //通知觀察者去更新 notify() { this.subs.forEach(w => w.update()) }}​​​// 定義一個Observer類通過gettr,setter實現數據的監聽綁定class Observer { constructor(data) { this.observer(data) } // 定義函數解析data,實現數據劫持 observer (data) { if(data && typeof data === ’object’) { // 是對象遍歷對象寫入getter,setter方法 Reflect.ownKeys(data).forEach(key => {this.defineReactive(data, key, data[key]); }) } } // 數據劫持方法 defineReactive(obj,key, value) { // 遞歸遍歷 this.observer(data) // 實例化一個dep對象 const dep = new Dep() // 通過ES5的API實現數據劫持 Object.defineProperty(obj, key, { enumerable: true, configurable: false, get() {// 當讀當前值的時候,會觸發。// 訂閱數據變化時,往Dep中添加觀察者Dep.target && dep.addSub(Dep.target)return value }, set: (newValue) => {// 對新數據進行劫持監聽this.observer(newValue)if(newValue !== value) { value = newValue}// 告訴dep通知變化dep.notify() } }) }}

三、總結

其實復雜的地方有三點:

1、指令解析的各種操作有點復雜饒人,其中包含DOM的基本操作和一些ES中的API使用。但是你靜下心去讀去想,肯定是能理順的。

2、數據劫持中Dep的理解,一是收集觀察者的集合,二是連接Observer和watcher的橋梁。

3、觀察者是什么時候進行綁定的?又是如何工作實現了數據驅動視圖,視圖驅動數據驅動視圖的。

在gitHub上有上述源碼地址,歡迎clone打樁嘗試,還請不要吝嗇一個小星星喲!

以上就是詳解Vue中的MVVM原理和實現方法的詳細內容,更多關于Vue中的MVVM的資料請關注好吧啦網其它相關文章!

標簽: Vue
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
亚洲青青久久| 亚洲欧美日韩精品一区二区| 日韩精品一级二级| 久久高清精品| 激情视频网站在线播放色| 精品国产精品国产偷麻豆 | 亚洲一区日本| 亚洲一区二区三区在线免费| 一区二区不卡| 玖玖精品视频| 亚洲三区欧美一区国产二区| 日韩激情视频网站| 国产劲爆久久| 免费观看亚洲| 欧美 日韩 国产一区二区在线视频| 欧美日韩国产高清电影| 好吊日精品视频| 蜜臀av国产精品久久久久| 欧美日韩精品一区二区三区在线观看| 欧美日韩亚洲国产精品| 97人人精品| 香蕉久久久久久久av网站| 国产亚洲人成a在线v网站| 国产拍在线视频| 亚洲精品1区2区| 视频国产精品| 日韩精品中文字幕第1页| 爽爽淫人综合网网站| 国产精品一线| 久久久久久久久久久9不雅视频| 亚洲精品乱码| 激情六月综合| 欧美日韩1区| 九一国产精品| 日韩伦理一区| 国产精品videossex久久发布| 国产一区日韩一区| 欧美黑人巨大videos精品| 99国产精品免费视频观看| 免费在线观看成人| 日韩88av| 欧美精品影院| 乱人伦精品视频在线观看| 国产精品免费99久久久| 免费人成网站在线观看欧美高清| 精品国产乱码久久久| 欧美亚洲网站| 亚洲精选av| 深夜视频一区二区| 国产精品2区| 亚州国产精品| 亚洲欧美成人综合| 99精品网站| 日韩精品诱惑一区?区三区| 91精品国产自产精品男人的天堂| 欧美gv在线| 97se综合| 中文在线а√在线8| 国产中文欧美日韩在线| 国产日韩欧美三级| 日本va欧美va精品| 亚洲精品影视| 亚洲影视一区二区三区| 老牛影视一区二区三区| 尤物在线精品| 亚洲免费成人| 亚洲欧美日韩国产一区| 视频一区视频二区中文字幕| 玖玖玖国产精品| 免费在线观看日韩欧美| 亚洲欧美日韩一区在线观看| 久久最新视频| 一区二区不卡| 国产欧美日韩精品一区二区免费 | 蜜臀va亚洲va欧美va天堂| 日韩高清三区| 久久精品国产网站| 九九精品调教| 国产精品呻吟| 久久黄色影视| 日韩中文首页| 天堂日韩电影| 中日韩男男gay无套| 免费在线观看视频一区| 日本欧美在线| 超碰超碰人人人人精品| 好看不卡的中文字幕| 亚洲精选91| 国产 日韩 欧美 综合 一区| 婷婷成人在线| 日韩精品亚洲一区二区三区免费| 久久97视频| 亚洲欧美日韩高清在线| 日本不卡一区二区三区| 黄色网一区二区| 美女精品在线| 国产精品国码视频| 欧美不卡在线| 精品五月天堂| 伊人久久一区| 久久久久久自在自线| 亚洲免费一区二区| 精品国产日韩欧美精品国产欧美日韩一区二区三区 | 日韩中文字幕亚洲一区二区va在线| 国产激情精品一区二区三区| 99视频一区| 久久国际精品| 日韩一区精品视频| 久久精品成人| 国产精品视频一区视频二区| 欧美日韩国产在线一区| 精品国产美女a久久9999| 综合亚洲自拍| 蜜臀av一区二区在线免费观看 | 激情婷婷亚洲| 精品精品久久| 久久福利在线| 欧美天堂在线| 欧美日韩一区二区三区在线电影| 秋霞影院一区二区三区 | 亚洲精品一区二区妖精| 六月婷婷综合| 国产一区二区三区精品在线观看| 日本成人在线不卡视频| 免费观看在线色综合| 国产毛片久久| 亚洲v天堂v手机在线| 久久aⅴ国产紧身牛仔裤| 欧美在线资源| 久久av在线| 婷婷综合国产| 日韩精品欧美精品| 91欧美日韩在线| 亚洲精品一二| 日本久久二区| 美女av一区| 日韩国产欧美一区二区| 成人羞羞视频播放网站| 亚洲性色av| 999在线观看精品免费不卡网站| 亚洲v在线看| 一区二区自拍| 免费视频一区二区| 国产精品成人一区二区网站软件| 91亚洲精品在看在线观看高清| 国产精品羞羞答答在线观看| 手机在线电影一区| 午夜欧美在线| 性欧美长视频| 国产欧美丝祙| 日韩一区二区中文| 久久国产精品毛片| 欧美精品影院| 国产精品99一区二区三| 在线亚洲一区| 精品国产日韩欧美精品国产欧美日韩一区二区三区| 成人亚洲一区| 国产亚洲福利| 国产传媒在线观看| 免费在线观看日韩欧美| 精品久久99| 视频一区欧美精品| 久久亚洲人体| 亚洲五月婷婷| 人人爱人人干婷婷丁香亚洲| 日韩在线短视频| 日韩欧美在线精品| 热三久草你在线| 国产精品九九| 亚洲综合中文| 激情综合自拍| 精品一区二区三区亚洲| 免费人成网站在线观看欧美高清| 成人国产精品一区二区网站| 婷婷精品久久久久久久久久不卡| 香蕉成人av| 精品午夜av| 国产精品一线| 中文字幕成人| 欧美日韩精品一区二区视频| 国产精品三级| 亚洲精品麻豆| 欧美午夜精彩| 亚洲爱爱视频| 亚洲天堂一区二区| 特黄毛片在线观看| 国产成人精品一区二区三区免费 | 亚洲精品国产偷自在线观看| 久久精品亚洲| 国产高清视频一区二区| 国产亚洲一区| 在线亚洲一区| 久久亚洲影院| 婷婷五月色综合香五月| 亚洲我射av| 亚洲一区二区小说| 亚州国产精品| 日韩欧美激情电影| 国产精品17p| 精品一区二区三区视频在线播放|