专栏名称: Fundebug
Fundebug为JavaScript、微信小程序及Node.js开发团队提供专业的线上代码bug监控和智能分析服务。
目录
相关文章推荐
前端大全  ·  从 DeepSeek 看25年前端的一个小趋势 ·  昨天  
前端早读课  ·  【招聘】字节跳动客服平台招高级前端开发工程师 ·  20 小时前  
歸藏的AI工具箱  ·  终于有给设计师用的 Cursor 了 ·  昨天  
歸藏的AI工具箱  ·  终于有给设计师用的 Cursor 了 ·  昨天  
前端早读课  ·  【第3454期】如何用语音学习编程的 ·  昨天  
51好读  ›  专栏  ›  Fundebug

教你撸一个简单的Vue

Fundebug  · 公众号  · 前端  · 2018-10-10 10:43

正文

摘要: 你知道怎样撸一个Vue吗?

  • 原文:JavaScript之实现一个简单的Vue

  • 地址:https://segmentfault.com/a/1190000016365102

  • 作者:wclimb

Fundebug经授权转载,版权归原作者所有。

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

Object.defineProperty()

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

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文档。

流程图

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: '我是粗体',
   },
   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 来监听数据,初始化需要订阅的数据。

把需要订阅的数据到 push watcherTask 里,等到时候需要更新的时候就可以批量更新数据了下面就是;

遍历订阅池,批量更新视图。

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






请到「今天看啥」查看全文