成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

擼個簡單的MVVM框架

imingyu / 549人閱讀

摘要:所以無需太過介懷是實現(xiàn)的單向或雙向綁定。響應(yīng)事件瀏覽器變更事件事件執(zhí)行或數(shù)據(jù)劫持則是采用數(shù)據(jù)劫持結(jié)合發(fā)布者訂閱者模式的方式,通過來劫持各個屬性的,,在數(shù)據(jù)變動時發(fā)布消息給訂閱者,觸發(fā)相應(yīng)的監(jiān)聽回調(diào)。

剖析Vue實現(xiàn)原理 - 如何實現(xiàn)雙向綁定mvvm

本文能幫你做什么?
1、了解vue的雙向數(shù)據(jù)綁定原理以及核心代碼模塊
2、緩解好奇心的同時了解如何實現(xiàn)雙向綁定
幾種實現(xiàn)雙向綁定的做法

目前幾種主流的mvc(vm)框架都實現(xiàn)了單向數(shù)據(jù)綁定,而我所理解的雙向數(shù)據(jù)綁定無非就是在單向綁定的基礎(chǔ)上給可輸入元素(input、textare等)添加了change(input)事件,來動態(tài)修改model和 view,并沒有多高深。所以無需太過介懷是實現(xiàn)的單向或雙向綁定。

實現(xiàn)數(shù)據(jù)綁定的做法有大致如下幾種:

發(fā)布者-訂閱者模式(backbone.js)

臟值檢查(angular.js)

數(shù)據(jù)劫持(vue.js)

發(fā)布者-訂閱者模式: 一般通過sub, pub的方式實現(xiàn)數(shù)據(jù)和視圖的綁定監(jiān)聽,更新數(shù)據(jù)方式通常做法是 vm.set("property", value),不太熟悉去問一下度娘

這種方式現(xiàn)在畢竟太low了,我們更希望通過 vm.property = value這種方式更新數(shù)據(jù),同時自動更新視圖,于是有了下面兩種方式

臟值檢查: angular.js 是通過臟值檢測的方式比對數(shù)據(jù)是否有變更,來決定是否更新視圖,最簡單的方式就是通過 setInterval() 定時輪詢檢測數(shù)據(jù)變動,當(dāng)然Google不會這么low,angular只有在指定的事件觸發(fā)時進入臟值檢測,大致如下:

DOM事件,譬如用戶輸入文本,點擊按鈕等。( ng-click )

XHR響應(yīng)事件 ( $http )

瀏覽器Location變更事件 ( $location )

Timer事件( $timeout , $interval )

執(zhí)行 $digest() 或 $apply()

數(shù)據(jù)劫持: vue.js 則是采用數(shù)據(jù)劫持結(jié)合發(fā)布者-訂閱者模式的方式,通過Object.defineProperty()來劫持各個屬性的settergetter,在數(shù)據(jù)變動時發(fā)布消息給訂閱者,觸發(fā)相應(yīng)的監(jiān)聽回調(diào)。

思路整理

已經(jīng)了解到vue是通過數(shù)據(jù)劫持的方式來做數(shù)據(jù)綁定的,其中最核心的方法便是通過Object.defineProperty()來實現(xiàn)對屬性的劫持,達到監(jiān)聽數(shù)據(jù)變動的目的,無疑這個方法是本文中最重要、最基礎(chǔ)的內(nèi)容之一,如果不熟悉defineProperty,猛戳這里 整理了一下,要實現(xiàn)mvvm的雙向綁定,就必須要實現(xiàn)以下幾點:

實現(xiàn)一個數(shù)據(jù)監(jiān)聽器Observer,能夠?qū)?shù)據(jù)對象的所有屬性進行監(jiān)聽,如有變動可拿到最新值并通知訂閱者

實現(xiàn)一個指令解析器Compile,對每個元素節(jié)點的指令進行掃描和解析,根據(jù)指令模板替換數(shù)據(jù),以及綁定相應(yīng)的更新函數(shù)

實現(xiàn)一個Watcher,作為連接Observer和Compile的橋梁,能夠訂閱并收到每個屬性變動的通知,執(zhí)行指令綁定的相應(yīng)回調(diào)函數(shù),從而更新視圖

mvvm入口函數(shù),整合以上三者

不多贅述,一言不合就上圖

大家可去下載去具體文件里面看,我寫了詳盡的注釋,每個模塊的功能,分工,每個方法任務(wù),等等

上圖為小編我根據(jù)自己的理解后重新繪制,本打算繪制再細一些,感覺會讓人理解更復(fù)雜而后就有了上圖,代碼中如果問題,歡迎指正,一起學(xué)習(xí),你們的start是小編的動力

下面為具體代碼實現(xiàn),為了大家方便我還是粘貼在readme里面,每個文件不多說了,前面做了文案及腦圖思路梳理,文件里我也了詳盡的注釋

MVVM.html



  
  
  
  mvvm
  
  
  
  
  


  
  
{{msg}}
MVVM.js
/**
 * -----------------------------------------------------
 * 1、實現(xiàn)數(shù)據(jù)代理
 * 2、模版解析
 * 3、劫持監(jiān)所有的屬性
 * -----------------------------------------------------
 */
class MVVM {
  /**
   *Creates an instance of MVVM.
   * @param {*} options 當(dāng)前實例傳遞過來的參數(shù)
   * @memberof MVVM
   */
  constructor(options){
    this.$opt = options|| {}
    this.$data = options.data;
    // 實現(xiàn)數(shù)據(jù)代理
    Object.keys(this.$data).forEach((key)=>{
      this._proxyData(key)
    })
    // 劫持監(jiān)所有的屬性
    observe(this.$data,this)
    // 模版編譯
    new Compile(options.el || document.body,this)
  }
  _proxyData(key){
    Object.defineProperty(this,key,{
      configurable:false,
      enumerable:true,
      get(){
        return this.$data[key]
      },
      set(newVal){
        this.$data[key] = newVal
      }
    })
  }
}
Observer.js
/**
 * -----------------------------------------------------
 * 1、實現(xiàn)一個數(shù)據(jù)監(jiān)聽器Observer
 * 2、通知和添加訂閱者
 * -----------------------------------------------------
 */
class Observer {
  /**
   *Creates an instance of Observer.
   * @param {*} data 需要劫持監(jiān)聽的數(shù)據(jù)
   * @memberof Observer
   */
  constructor(data){
    this.$data = data || {}
    this.init()
  }
  init(){
    Object.keys(this.$data).forEach(key=>{
      this.defineReative(key,this.$data[key])
    })
  }
  defineReative(key,val){
    // 創(chuàng)建發(fā)布者-訂閱者
    let dep = new Dep()
    // 再去觀察子對象
    observe(val)
    Object.defineProperty(this.$data,key,{
      configurable:false,
      enumerable:true,
      get(){
        // 添加訂閱者
        Dep.target && dep.addSub(Dep.target)
        return val
      },
      set(newVal){
        if( newVal == val ) return false;
        val = newVal
        // 新的值是object的話,進行監(jiān)聽
        observe(newVal)
        // 通知訂閱者
        dep.notfiy()
      }
    })
  }
}
/**
 * 是否進行劫持監(jiān)聽
 *
 * @param {*} value 監(jiān)聽對象
 * @param {*} vm 當(dāng)前實例
 * @returns 返回 監(jiān)聽實例
 */
function observe(value, vm) {
  if (!value || typeof value !== "object") {
      return;
  }
  return new Observer(value);
};
class Dep{
  constructor(){
    this.subs = []
  }
  /**
   *維護訂閱者數(shù)組
   *
   * @param {*} sub 訂閱實例
   * @memberof Dep
   */
  addSub(sub){
    this.subs.push(sub)
  }
  notfiy(){
    this.subs.forEach(sub=>{
      // 通知數(shù)據(jù)更新
      sub.update()
    })
  }
}
Compile.js
/**
 * -----------------------------------------------------
 * 1、取真實dom節(jié)點
 * 2、我們fragment 創(chuàng)建文檔碎片,將真是dmo,移動指緩存
 * 3、編譯虛擬dom,解析模版語法
 * 4、回填至真是dom,實現(xiàn)模版語法解析,更新試圖
 * -----------------------------------------------------
 */
class Compile{
  /**
   * 
   *Creates an instance of Compile.
   * @param {*} el dmo選擇器
   * @param {*} vm 當(dāng)前實例
   * @memberof Compile
   */
  constructor(el,vm){
    this.$vm = vm;
    this.$el = this.isElementNode(el) ? el : document.querySelector(el)
    if(this.$el){
      this.$fragment = this.node2Fragment(this.$el)
      this.init()
      this.$el.appendChild(this.$fragment)
    }
  }
  init(){
    this.compileElement(this.$fragment)
  }
  /**
   *
   * 編譯element
   * @param {*} el dmo節(jié)點
   * @memberof Compile
   */
  compileElement(el){
    // 1、取所有子節(jié)點
    let childNodes = el.childNodes
    // 2、循環(huán)子節(jié)點
    Array.from(childNodes).forEach((node)=>{
      // 判斷是文本節(jié)點還是dom節(jié)點
      if(this.isElementNode(node)){
        this.compileDom(node)
      }else if (this.isTextNode(node)){
        this.compileText(node)
      }
      // 判斷當(dāng)前節(jié)點是否有子節(jié)點,如果有,遞歸查找
      if(node.childNodes && node.childNodes.length){
        this.compileElement(node)
      }
    })
  }
  /**
   *
   * 編譯元素節(jié)點
   * @param {*} node 需要編譯的當(dāng)前節(jié)點
   * @memberof Compile
   */
  compileDom(node){
    // 取當(dāng)前節(jié)點的屬性集合
    let attrs = node.attributes
    // 循環(huán)屬性數(shù)組
    Array.from(attrs).forEach(attr => {
      let attrName = attr.name
      // 判斷當(dāng)前屬性是否是指令
      if(this.isDirective(attrName)){
        let [,dir] = attrName.split("-")
        let expr = attr.value
        //判斷當(dāng)前屬性是普通指令還是事件指令
        if(this.isEventDirective(dir)){
          compileUtil.eventHandler(node,expr,dir,this.$vm)
        }else{
          compileUtil[dir] && compileUtil[dir](node,expr,this.$vm)
        }
      }
    });
  }
  /**
   * 
   * 編譯文本節(jié)點
   * @param {*} node 需要編譯的當(dāng)前節(jié)點
   * @memberof Compile
   */
  compileText(node){
    var text = node.textContent;
    var reg = /{{(.*)}}/;
    if(reg.test(text)){
      compileUtil.text(node,RegExp.$1,this.$vm)
    }
  }
  /**
   * 判斷是否是元素節(jié)點
   *
   * @param {*} el 節(jié)點
   * @returns 是否
   * @memberof Compile
   */
  isElementNode(el){
    return el.nodeType == 1
  }
  /**
   * 過濾是否是指令
   *
   * @param {*} name 屬性名
   * @returns 是否
   * @memberof Compile
   */
  isDirective(name){
    return name.indexOf("v-") == 0
  }
  /**
   * 判斷是否是事件指令
   *
   * @param {*} dir 指令,on:click
   * @returns 是否
   * @memberof Compile
   */
  isEventDirective(dir){
    return dir.indexOf("on") == 0
  }
  /**
   * 判斷是否是文本節(jié)點
   *
   * @param {*} el 節(jié)點
   * @returns 是否
   * @memberof Compile
   */
  isTextNode(el){
    return el.nodeType == 3
  }
  /**
   * 將真實dom拷貝到內(nèi)存中
   *
   * @param {*} el 真實dom
   * @returns 文檔碎片
   * @memberof Compile
   */
  node2Fragment(el){
    let fragment = document.createDocumentFragment();
    let children
    while(children = el.firstChild){
      fragment.appendChild(el.firstChild)
    }
    return fragment
  }
}

// 指令處理工具
let compileUtil = {
  /**
   * 處理文本節(jié)點
   *
   * @param {*} node 當(dāng)前節(jié)點
   * @param {*} expr 表達式
   * @param {*} vm 當(dāng)前實例
   */
  text(node,expr,vm){
    this.buid(node,expr,vm,"text")
  },
  /**
   * 處理表單元素節(jié)點
   *
   * @param {*} node 當(dāng)前節(jié)點
   * @param {*} expr 表達式
   * @param {*} vm 當(dāng)前實例
   */
  model(node,expr,vm){
    this.buid(node,expr,vm,"model")
    var me = this,
    val = this.getVMVal(vm, expr);
    node.addEventListener("input", function(e) {
        var newValue = e.target.value;
        if (val === newValue) {
            return;
        }

        me.setVMVal(vm, expr, newValue);
        val = newValue;
    });
  },
  /**
   * 事件處理
   *
   * @param {*} node 當(dāng)前節(jié)點
   * @param {*} expr 表達式
   * @param {*} dir 指令
   * @param {*} vm 當(dāng)前實例
   */
  eventHandler(node,expr,dir,vm){
    let [,eventType] = dir.split(":");
    let fn = vm.$opt.methods && vm.$opt.methods[expr]
    if(eventType && fn){
      node.addEventListener(eventType,fn.bind(vm),false)
    }
  },
  /**
   * 綁定事件統(tǒng)一處理方法抽離,添加watcher
   *
   * @param {*} node 當(dāng)前節(jié)點
   * @param {*} expr 表達式
   * @param {*} vm 當(dāng)前實例
   * @param {*} dir 指令
   */
  buid(node,expr,vm,dir){
    let updateFn = update[dir+"Update"]
    updateFn && updateFn(node,this.getVMVal(vm,expr))

    new Watcher(vm, expr, function(value, oldValue) {
      updateFn && updateFn(node, value, oldValue);
  });
  },
  /**
   * 獲取表達式代表的值
   *
   * @param {*} vm 當(dāng)前實例
   * @param {*} expr 表達式
   * @returns
   */
  getVMVal(vm,expr){
    // return vm[expr] 要考慮,a.b.c的情況
    let exp = expr.split(".");
    let val = vm
    exp.forEach((k)=>{
      val = val[k]
    })
    return val
  },
  /**
   * 設(shè)置更新數(shù)據(jù)里對應(yīng)的表達式的值
   *
   * @param {*} vm
   * @param {*} expr
   * @param {*} newValue
   */
  setVMVal(vm,expr,newValue){
    let exp = expr.split(".");
    let val = vm
    exp.forEach((key,i)=>{
      if(i
Watcher.js
/**
 * -----------------------------------------------------
 * 1、實現(xiàn)一個Watcher,作為連接Observer和Compile的橋梁
 * 2、通知和添加訂閱者
 * -----------------------------------------------------
 */
class Watcher {
  /**
   *Creates an instance of Watcher.
   * @param {*} vm 當(dāng)前實例
   * @param {*} expOrFn 表達式
   * @param {*} cb 更新回調(diào)用
   * @memberof Watcher
   */
  constructor(vm,expOrFn,cb){
    this.$vm = vm
    this.$expOrFn = expOrFn
    this.$cb = cb
    this.value = this.get()
  }
  get(){
    // 添加訂閱者
    Dep.target = this;
    // let dep = new Dep()
    // 去modal中取值,這個時候必然會觸發(fā)defineProperty的getter,真正的push訂閱者
    let value = this.getVMVal(this.$vm,this.$expOrFn)
    // 用完了,重置回去
    Dep.target = null
    return value
  }
  /**
   * 取modal里的值
   *
   * @param {*} vm 當(dāng)前實例
   * @param {*} expr 表達式
   * @returns 返回指
   * @memberof Watcher
   */
  getVMVal(vm,expr){
    // return vm[expr] 要考慮,a.b.c的情況
    let exp = expr.split(".");
    let val = vm
    exp.forEach((k)=>{
      val = val[k]
    })
    return val
  }
  // 對外暴露的跟新方法,比較新老值,得到訂閱通知進行更新
  update(){
    let oldVal = this.value;
    let newVal = this.getVMVal(this.$vm,this.$expOrFn)
    if (newVal !== oldVal) {
        this.value = newVal;
        this.$cb(newVal, oldVal);
    }
  }
}

最后感謝您的閱讀

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/96407.html

相關(guān)文章

  • 用typescript擼個前端框架InDiv

    摘要:暫時沒有指令和。當(dāng)前模塊內(nèi)的組件可以使用來自根模塊和當(dāng)前模塊的任何服務(wù)及組件,也可以使用被導(dǎo)入模塊中導(dǎo)出的組件。作為一個前端菜雞,還是在深知自己眾多不足以及明白好記性不如爛筆頭的道理下,多造輪子總歸不會錯的。 有個同事跟我說:需求還是不夠多,都有時間造輪子了。。。 前言 這個輪子從18年4月22造到18年10月12日,本來就是看了一個文章講前端框架的路由實現(xiàn)原理之后,想試著擼一個路由試...

    liangzai_cool 評論0 收藏0
  • 擼個查詢物流小程序,歡迎體驗

    摘要:微信搜索小程序查一查物流,或者掃一掃下圖,歡迎來回復(fù)分享哦。小程序用框架開發(fā)的,方便快捷,寫法類似,支持相關(guān)操作,已可以引入包,不過在微信開發(fā)者工具有以下注意事項。對應(yīng)關(guān)閉轉(zhuǎn)選項,關(guān)閉。對應(yīng)關(guān)閉上傳代碼時樣式自動補全選項,關(guān)閉。 微信搜索小程序 查一查物流,或者掃一掃下圖,歡迎來回復(fù)分享哦。 showImg(https://segmentfault.com/img/bVbiR2p?w=...

    張巨偉 評論0 收藏0
  • 擼個查詢物流小程序,歡迎體驗

    摘要:微信搜索小程序查一查物流,或者掃一掃下圖,歡迎來回復(fù)分享哦。小程序用框架開發(fā)的,方便快捷,寫法類似,支持相關(guān)操作,已可以引入包,不過在微信開發(fā)者工具有以下注意事項。對應(yīng)關(guān)閉轉(zhuǎn)選項,關(guān)閉。對應(yīng)關(guān)閉上傳代碼時樣式自動補全選項,關(guān)閉。 微信搜索小程序 查一查物流,或者掃一掃下圖,歡迎來回復(fù)分享哦。 showImg(https://segmentfault.com/img/bVbiR2p?w=...

    JeOam 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<