JavaScript之实现一个简单的Vue示例


Posted in Javascript onJanuary 17, 2019

vue的使用相信大家都很熟练了,使用起来简单。但是大部分人不知道其内部的原理是怎么样的,今天我们就来一起实现一个简单的vue

Object.defineProperty()

实现之前我们得先看一下Object.defineProperty的实现,因为vue主要是通过数据劫持来实现的,通过getset来完成数据的读取和更新。

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获取数据,通过set监听到数据变化执行相应操作,还是不明白的话可以去看看Object.defineProperty文档。

流程图

JavaScript之实现一个简单的Vue示例

html代码结构

<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调用

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

Vue结构

class Vue{
    constructor(){}
    proxyData(){}
    observer(){}
    compile(){}
    compileText(){}
  }
  class Watcher{
    constructor(){}
    update(){}
  }
  • Vue constructor 构造函数主要是数据的初始化
  • proxyData 数据代理
  • observer 劫持监听所有数据
  • 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 = {}; // 需要监听的任务列表
      this.observer(data); // 初始化劫持监听所有数据
      this.compile(this.$el); // 解析dom
    }
  }

上面主要是初始化操作,针对传过来的数据进行处理

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 劫持监听

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来监听数据,初始化需要订阅的数据。

把需要订阅的数据到pushwatcherTask里,等到时候需要更新的时候就可以批量更新数据了。?下面就是;
遍历订阅池,批量更新视图。

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]
          }
        })
      }
    }
  }

这里代码比较多,我们拆分看你就会觉得很简单了

首先我们先遍历el元素下面的所有子节点,node.nodeType === 3 的意思是当前元素是文本节点,node.nodeType === 1 的意思是当前元素是元素节点。因为可能有的是纯文本的形式,如纯双花括号就是纯文本的文本节点,然后通过判断元素节点是否还存在子节点,如果有的话就递归调用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')
}

上面这个首先判断node节点上是否有v-html这种指令,如果存在的话,我们就发布订阅,怎么发布订阅呢?只需要把当前需要订阅的数据pushwatcherTask里面,然后到时候在设置值的时候就可以批量更新了,实现双向数据绑定,也就是下面的操作

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

然后push的值是一个Watcher的实例,首先他new的时候会先执行一次,执行的操作就是去把纯双花括号 -> 1,也就是说把我们写好的模板数据更新到模板视图上。
最后把当前元素属性剔除出去,我们用Vue的时候也是看不到这种指令的,不剔除也不影响

至于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]
  }
}

之前发布订阅之后走了这里面的操作,意思就是把当前元素如:node.innerHTML = '这是data里面的值'、node.value = '这个是表单的数据'

那么我们为什么不直接去更新呢,还需要update做什么,不是多此一举吗?
其实update记得吗?我们在订阅池里面需要批量更新,就是通过调用Watcher原型上的update方法。

效果

在线效果地址,大家可以浏览器看一下效果,由于本人太懒了,gif效果图就先不放了,哈哈??

完整代码

完整代码已经放到github上了 -> MyVue

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
Prototype Class对象学习
Jul 19 Javascript
学习ExtJS TextField常用方法
Oct 07 Javascript
firefox下frameset取不到值的解决方法
Sep 06 Javascript
javascript动态加载二
Aug 22 Javascript
JavaScript中九种常用排序算法
Sep 02 Javascript
推荐4个原生javascript常用的函数
Jan 12 Javascript
jQuery使用中可能被XSS攻击的一些危险环节提醒
May 24 Javascript
AngularJs入门教程之环境搭建+创建应用示例
Nov 01 Javascript
vue基础之事件简写、事件对象、冒泡、默认行为、键盘事件实例分析
Mar 11 Javascript
小试小程序云开发(小结)
Jun 06 Javascript
JS+CSS实现3D切割轮播图
Mar 21 Javascript
vue+axios 拦截器实现统一token的案例
Sep 11 Javascript
如何能分清npm cnpm npx nvm
Jan 17 #Javascript
JavaScript设计模式之装饰者模式实例详解
Jan 17 #Javascript
npm 常用命令详解(小结)
Jan 17 #Javascript
JavaScript设计模式之享元模式实例详解
Jan 17 #Javascript
vuex页面刷新后数据丢失的方法
Jan 17 #Javascript
jquery获取img的src值实例介绍
Jan 16 #jQuery
js中int和string数据类型互相转化实例
Jan 16 #Javascript
You might like
PHP+MYSQL的文章管理系统(一)
2006/10/09 PHP
PHP类的使用 实例代码讲解
2009/12/28 PHP
PHP 编写大型网站问题集
2010/05/07 PHP
PHP递归返回值时出现的问题解决办法
2013/02/19 PHP
php通过分类列表产生分类树数组的方法
2015/04/20 PHP
php 数组字符串搜索array_search技巧
2016/07/05 PHP
JavaScript之HTMLCollection接口代码
2011/04/27 Javascript
jQuery EasyUI API 中文文档 - Pagination分页
2011/09/29 Javascript
javascript的offset、client、scroll使用方法详解
2012/12/25 Javascript
javascript 树形导航菜单实例代码
2013/08/13 Javascript
不到30行JS代码实现Excel表格的方法
2014/11/15 Javascript
js本地图片预览实现代码
2016/10/09 Javascript
微信小程序  wx.request合法域名配置详解
2016/11/23 Javascript
详解vue中computed 和 watch的异同
2017/06/30 Javascript
vue中使用cropperjs的方法
2018/03/01 Javascript
Vue + better-scroll 实现移动端字母索引导航功能
2018/05/07 Javascript
微信小程序+云开发实现欢迎登录注册
2019/05/24 Javascript
vue跳转方式(打开新页面)及传参操作示例
2020/01/26 Javascript
使用vue引入maptalks地图及聚合效果的实现
2020/08/10 Javascript
python异常和文件处理机制详解
2016/07/19 Python
Python如何快速实现分布式任务
2017/07/06 Python
Linux(Redhat)安装python3.6虚拟环境(推荐)
2018/05/05 Python
python实现公司年会抽奖程序
2019/01/22 Python
python实现kNN算法识别手写体数字的示例代码
2019/08/16 Python
IE9对HTML5中部分属性不支持的原因分析
2014/10/15 HTML / CSS
触发器(trigger)的功能都有哪些?写出一个触发器的例子
2012/09/17 面试题
某同学的自我鉴定范文
2013/12/26 职场文书
税务干部鉴定材料
2014/02/11 职场文书
委托书格式
2014/08/01 职场文书
导游词格式
2015/02/13 职场文书
银行招聘自荐信
2015/03/06 职场文书
卫生院艾滋病宣传活动总结
2015/05/09 职场文书
2015国庆66周年宣传语
2015/07/14 职场文书
三严三实·严以用权心得体会
2016/01/12 职场文书
老生常谈 使用 CSS 实现三角形的技巧(多种方法)
2021/04/13 HTML / CSS
Python字符串的转义字符
2022/04/07 Python