欧美一级特黄大片做受成人-亚洲成人一区二区电影-激情熟女一区二区三区-日韩专区欧美专区国产专区

JavaScript實(shí)現(xiàn)一個(gè)簡單的Vue-創(chuàng)新互聯(lián)

JavaScript實(shí)現(xiàn)一個(gè)簡單的Vue?相信很多沒有經(jīng)驗(yàn)的人對此束手無策,為此本文總結(jié)了問題出現(xiàn)的原因和解決方法,通過這篇文章希望你能解決這個(gè)問題。

江永ssl適用于網(wǎng)站、小程序/APP、API接口等需要進(jìn)行數(shù)據(jù)傳輸應(yīng)用場景,ssl證書未來市場廣闊!成為創(chuàng)新互聯(lián)的ssl證書銷售渠道,可以享受市場價(jià)格4-6折優(yōu)惠!如果有意向歡迎電話聯(lián)系或者加微信:028-86922220(備注:SSL證書合作)期待與您的合作!

Object.defineProperty()

實(shí)現(xiàn)之前我們得先看一下Object.defineProperty的實(shí)現(xiàn),因?yàn)関ue主要是通過數(shù)據(jù)劫持來實(shí)現(xiàn)的,通過get、set來完成數(shù)據(jù)的讀取和更新。

var obj = {name:'wclimb'}
var age = 24
Object.defineProperty(obj,'age',{
  enumerable: true, // 可枚舉
  configurable: false, // 不能再define
  get () {
    return age
  },
  set (newVal) {
    console.log('我改變了',age +' -> '+newVal);
    age = newVal
  }
})

> obj.age
> 24

> obj.age = 25;
> 我改變了 24 -> 25
> 25

從上面可以看到通過get獲取數(shù)據(jù),通過set監(jiān)聽到數(shù)據(jù)變化執(zhí)行相應(yīng)操作,還是不明白的話可以去看看Object.defineProperty文檔。

流程圖

JavaScript實(shí)現(xiàn)一個(gè)簡單的Vue

html代碼結(jié)構(gòu)

<div id="wrap">
  <p v-html="test"></p>
  <input type="text" v-model="form">
  <input type="text" v-model="form">
  <button @click="changeValue">改變值</button>
  {{form}}
</div>

js調(diào)用

 new Vue({
    el: '#wrap',
    data:{
      form: '這是form的值',
      test: '<strong>我是粗體</strong>',
    },
    methods:{
      changeValue(){
        console.log(this.form)
        this.form = '值被我改變了,氣不氣?'
      }
    }
  })

Vue結(jié)構(gòu)

  class Vue{
    constructor(){}
    proxyData(){}
    observer(){}
    compile(){}
    compileText(){}
  }
  class Watcher{
    constructor(){}
    update(){}
  }
  • Vue constructor 構(gòu)造函數(shù)主要是數(shù)據(jù)的初始化

  • proxyData 數(shù)據(jù)代理

  • observer 劫持監(jiān)聽所有數(shù)據(jù)

  • compile 解析dom

  • compileText 解析dom里處理純雙花括號的操作

  • Watcher 更新視圖操作

Vue constructor 初始化

  class Vue{
    constructor(options = {}){
      this.$el = document.querySelector(options.el);
      let data = this.data = options.data; 
      // 代理data,使其能直接this.xxx的方式訪問data,正常的話需要this.data.xxx
      Object.keys(data).forEach((key)=> {
        this.proxyData(key);
      });
      this.methods = options.methods // 事件方法
      this.watcherTask = {}; // 需要監(jiān)聽的任務(wù)列表
      this.observer(data); // 初始化劫持監(jiān)聽所有數(shù)據(jù)
      this.compile(this.$el); // 解析dom
    }
  }

上面主要是初始化操作,針對傳過來的數(shù)據(jù)進(jìn)行處理

proxyData 代理data

class Vue{
    constructor(options = {}){
      ......
    }
    proxyData(key){
      let that = this;
      Object.defineProperty(that, key, {
        configurable: false,
        enumerable: true,
        get () {
          return that.data[key];
        },
        set (newVal) {
          that.data[key] = newVal;
        }
      });
    }
  }

上面主要是代理data到最上層,this.xxx的方式直接訪問data

observer 劫持監(jiān)聽

class Vue{
    constructor(options = {}){
      ......
    }
    proxyData(key){
      ......
    }
    observer(data){
      let that = this
      Object.keys(data).forEach(key=>{
        let value = data[key]
        this.watcherTask[key] = []
        Object.defineProperty(data,key,{
          configurable: false,
          enumerable: true,
          get(){
            return value
          },
          set(newValue){
            if(newValue !== value){
              value = newValue
              that.watcherTask[key].forEach(task => {
                task.update()
              })
            }
          }
        })
      })
    }
  }

同樣是使用Object.defineProperty來監(jiān)聽數(shù)據(jù),初始化需要訂閱的數(shù)據(jù)。

把需要訂閱的數(shù)據(jù)到pushwatcherTask里,等到時(shí)候需要更新的時(shí)候就可以批量更新數(shù)據(jù)了。?下面就是;
遍歷訂閱池,批量更新視圖。

  set(newValue){
    if(newValue !== value){
      value = newValue
      // 批量更新視圖
      that.watcherTask[key].forEach(task => {
        task.update()
      })
    }
  }

compile 解析dom

class Vue{
    constructor(options = {}){
      ......
    }
    proxyData(key){
      ......
    }
    observer(data){
      ......
    }
    compile(el){
      var nodes = el.childNodes;
      for (let i = 0; i < nodes.length; i++) {
        const node = nodes[i];
        if(node.nodeType === 3){
          var text = node.textContent.trim();
          if (!text) continue;
          this.compileText(node,'textContent')        
        }else if(node.nodeType === 1){
          if(node.childNodes.length > 0){
            this.compile(node)
          }
          if(node.hasAttribute('v-model') && (node.tagName === 'INPUT' || node.tagName === 'TEXTAREA')){
            node.addEventListener('input',(()=>{
              let attrVal = node.getAttribute('v-model')
              this.watcherTask[attrVal].push(new Watcher(node,this,attrVal,'value'))
              node.removeAttribute('v-model')
              return () => {
                this.data[attrVal] = node.value
              }
            })())
          }
          if(node.hasAttribute('v-html')){
            let attrVal = node.getAttribute('v-html');
            this.watcherTask[attrVal].push(new Watcher(node,this,attrVal,'innerHTML'))
            node.removeAttribute('v-html')
          }
          this.compileText(node,'innerHTML')
          if(node.hasAttribute('@click')){
            let attrVal = node.getAttribute('@click')
            node.removeAttribute('@click')
            node.addEventListener('click',e => {
              this.methods[attrVal] && this.methods[attrVal].bind(this)()
            })
          }
        }
      }
    },
    compileText(node,type){
      let reg = /\{\{(.*?)\}\}/g, txt = node.textContent;
      if(reg.test(txt)){
        node.textContent = txt.replace(reg,(matched,value)=>{
          let tpl = this.watcherTask[value] || []
          tpl.push(new Watcher(node,this,value,type))
          if(value.split('.').length > 1){
            let v = null
            value.split('.').forEach((val,i)=>{
              v = !v ? this[val] : v[val]
            })
            return v
          }else{
            return this[value]
          }
        })
      }
    }
  }

這里代碼比較多,我們拆分看你就會(huì)覺得很簡單了

首先我們先遍歷el元素下面的所有子節(jié)點(diǎn),node.nodeType === 3 的意思是當(dāng)前元素是文本節(jié)點(diǎn),node.nodeType === 1 的意思是當(dāng)前元素是元素節(jié)點(diǎn)。因?yàn)榭赡苡械氖羌兾谋镜男问?,?code>純雙花括號就是純文本的文本節(jié)點(diǎn),然后通過判斷元素節(jié)點(diǎn)是否還存在子節(jié)點(diǎn),如果有的話就遞歸調(diào)用compile方法。下面重頭戲來了,我們拆開看:

if(node.hasAttribute('v-html')){
  let attrVal = node.getAttribute('v-html');
  this.watcherTask[attrVal].push(new Watcher(node,this,attrVal,'innerHTML'))
  node.removeAttribute('v-html')
}

上面這個(gè)首先判斷node節(jié)點(diǎn)上是否有v-html這種指令,如果存在的話,我們就發(fā)布訂閱,怎么發(fā)布訂閱呢?只需要把當(dāng)前需要訂閱的數(shù)據(jù)pushwatcherTask里面,然后到時(shí)候在設(shè)置值的時(shí)候就可以批量更新了,實(shí)現(xiàn)雙向數(shù)據(jù)綁定,也就是下面的操作

that.watcherTask[key].forEach(task => {
  task.update()
})

然后push的值是一個(gè)Watcher的實(shí)例,首先他new的時(shí)候會(huì)先執(zhí)行一次,執(zhí)行的操作就是去把純雙花括號 -> 1,也就是說把我們寫好的模板數(shù)據(jù)更新到模板視圖上。
最后把當(dāng)前元素屬性剔除出去,我們用Vue的時(shí)候也是看不到這種指令的,不剔除也不影響

至于Watcher是什么,看下面就知道了

Watcher

class Watcher{
  constructor(el,vm,value,type){
    this.el = el;
    this.vm = vm;
    this.value = value;
    this.type = type;
    this.update()
  }
  update(){
    this.el[this.type] = this.vm.data[this.value]
  }
}

看完上述內(nèi)容,你們掌握J(rèn)avaScript實(shí)現(xiàn)一個(gè)簡單的Vue的方法了嗎?如果還想學(xué)到更多技能或想了解更多相關(guān)內(nèi)容,歡迎關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道,感謝各位的閱讀!

當(dāng)前標(biāo)題:JavaScript實(shí)現(xiàn)一個(gè)簡單的Vue-創(chuàng)新互聯(lián)
網(wǎng)站鏈接:http://aaarwkj.com/article38/dopspp.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供軟件開發(fā)、營銷型網(wǎng)站建設(shè)虛擬主機(jī)、App開發(fā)、定制開發(fā)、網(wǎng)站制作

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來源: 創(chuàng)新互聯(lián)

外貿(mào)網(wǎng)站建設(shè)
国产三级在线观看视频| 黄片无毛欧美在线观看| 国产色视频一区在线观看| 蜜桃网站视频免费观看| 日韩国产推荐一区二区| 久久精品有码视频免费观看| 国产日韩精品一区二区三区在线| 日本成人一区二区三区视频| 婷婷国产成人精品一区二| 熟女人妻av五十路六十路| 亚洲成人影院中文字幕| 午夜情色视频在线观看| 九九久久亚洲av成人乱片| 亚洲精品不卡在线观看| 亚洲一区二区视频免费看| 亚洲五月婷婷久久综合| 日韩综合欧美激情另类| 日韩精品一区二区国产| 亚洲国产日本一区自拍| 97在线公开免费视频| 亚洲国产精品成人久久蜜臀| 超碰欧美性欧美最猛性| 日本精品中文字幕人妻| 巨乳人妻一区二区三区| 日本一区二区精美视频| 粉嫩av北条麻妃电影| 欧美日韩国内在线视频| 国产午夜在线观看免费视频| 亚洲男人的av天堂生活| 中文字幕日本精品人妻在线| 日本一区二区免费视频| 夫妻在线观看高清视频| 特别黄的日本免费视频| 欧美熟妇精品一区二区蜜桃| 中国女人内射91熟女| 中文字幕有码av海量| 国产精品日韩欧美久久久| 欧美一区二区高清不卡| 亚洲各类熟女们中文字幕| 伊人亚洲一区二区三区| 日本一区二区欧美在线|