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 相关文章推荐
JavaScript 全面解析各种浏览器网页中的JS 执行顺序
Feb 17 Javascript
JS定时刷新页面及跳转页面的方法
Jul 04 Javascript
JavaScript数据结构和算法之图和图算法
Feb 11 Javascript
javascript字符串替换函数如何一次性全部替换掉
Oct 30 Javascript
Javascript获取数组中的最大值和最小值的方法汇总
Jan 01 Javascript
如何使用Vuex+Vue.js构建单页应用
Oct 27 Javascript
详解React-Todos入门例子
Nov 08 Javascript
实例讲解Vue.js中router传参
Apr 22 Javascript
jQuery插件jsonview展示json数据
May 26 jQuery
VUE 实现滚动监听 导航栏置顶的方法
Sep 11 Javascript
vue2.0项目集成Cesium的实现方法
Jul 30 Javascript
layui 根据后台数据动态创建下拉框并同时默认选中的实例
Sep 02 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
symfony表单与页面实现技巧
2015/01/26 PHP
PHP回溯法解决0-1背包问题实例分析
2015/03/23 PHP
Yii2配置Nginx伪静态的方法
2017/05/05 PHP
javascript IE中的DOM ready应用技巧
2008/07/23 Javascript
JavaScript 学习笔记(四)
2009/12/31 Javascript
jQuery EasyUI API 中文文档 - Menu菜单
2011/10/03 Javascript
javascript 二进制运算技巧解析
2012/11/27 Javascript
Javascript算符的优先级介绍
2013/03/20 Javascript
Jquery实现点击切换图片并隐藏显示内容(2种方法实现)
2013/04/11 Javascript
jQuery实现鼠标经过提示信息的地图热点效果
2015/04/26 Javascript
基于JS实现的倒计时程序实例
2015/07/24 Javascript
JS实现漂亮的时间选择框效果
2016/08/20 Javascript
jQuery实现为LI列表前3行设置样式的方法【2种方法】
2016/09/04 Javascript
js将字符串中的每一个单词的首字母变为大写其余均为小写
2017/01/05 Javascript
微信小程序 共用变量值的实现
2017/07/12 Javascript
利用jQuery异步上传文件的插件用法详解
2017/07/19 jQuery
vue引入新版 vue-awesome-swiper插件填坑问题
2018/01/25 Javascript
详解react、redux、react-redux之间的关系
2018/04/11 Javascript
vue-video-player 通过自定义按钮组件实现全屏切换效果【推荐】
2018/08/29 Javascript
JS实现表单中点击小眼睛显示隐藏密码框中的密码
2020/04/13 Javascript
Django集成百度富文本编辑器uEditor攻略
2014/07/04 Python
详解Python中的正则表达式的用法
2015/04/09 Python
python基础入门学习笔记(Python环境搭建)
2016/01/13 Python
python socket网络编程之粘包问题详解
2018/04/28 Python
Python通过正则库爬取淘宝商品信息代码实例
2020/03/02 Python
Python存储读取HDF5文件代码解析
2020/11/25 Python
Ray-Ban雷朋美国官网:全球领先的太阳眼镜品牌
2016/07/20 全球购物
Michael Kors美国官网:美式奢侈生活风格的代表
2016/11/25 全球购物
Harman Audio官方商店:购买JBL、Harman Kardon、Infinity和AKG
2019/12/05 全球购物
亿阳信通股份有限公司笔试题(C#)
2016/03/04 面试题
特色蛋糕店创业计划书
2014/01/28 职场文书
校园联欢晚会主持词
2014/03/17 职场文书
超市食品安全承诺书
2015/04/29 职场文书
教师远程培训心得体会
2016/01/09 职场文书
2016年小学“感恩教师”主题队日活动总结
2016/04/01 职场文书
Go语言应该什么情况使用指针
2021/07/25 Golang