Vue源码学习记录

Vue源码学习记录

  • 准备工作
  • 源码分析
    • 数据代理(MVVM.js)
    • 模板解析(compile.js)
    • 数据劫持-->数据绑定
      • 1.数据绑定(model==>View):
      • 2.数据劫持
      • 3.四个重要对象
        • 1.Observer
        • 2. Dep(Depend)
        • 3. Compile
        • 4. Watcher
        • 5. 总结: dep与watcher的关系: 多对多
      • 4. 双向数据绑定
      • MVVM原理图

准备工作

用到一些js的api

  1. [].slice.call(lis): 将伪数组转换为真数组
  2. node.nodeType: 得到节点类型
  3. Object.defineProperty(obj, propName, {}): 给对象添加/修改属性(指定描述符)
    1. configurable: true/false 是否可以重新 define
    2. enumerable: true/false 是否可以枚举(for…in / keys())
    3. value: 指定初始值
    4. writable: true/false value 是否可以修改
    5. get: 回调函数, 用来得到当前属性值
    6. set: 回调函数, 用来监视当前属性值的变化
  4. Object.keys(obj): 得到对象自身可枚举的属性名的数组
  5. DocumentFragment: 文档碎片(高效批量更新多个节点)
  6. obj.hasOwnProperty(prop): 判断 prop 是否是 obj 自身的属性

源码分析

数据代理(MVVM.js)

1.通过一个对象代理对另一个对象中属性的操作(读/写)
2.通过vm对象来代理data对象中所有属性的操作
3.好处: 更方便的操作data中的数据
4.基本实现流程
	1). 通过Object.defineProperty()给vm添加与data对象的属性对应的属性描述符
	2). 所有添加的属性都包含getter/setter
	3). 在getter/setter内部去操作data中对应的属性数据
  • 数据代理的关键语法
// 对指定属性实现代理
  _proxy: function (key) {
    // 保存vm
    var me = this;
    // 给vm添加指定属性名的属性(使用属性描述)
    Object.defineProperty(me, key, {
      configurable: false, // 不能再重新定义
      enumerable: true, // 可以枚举
      // 当通过vm.name读取属性值时自动调用
      get: function proxyGetter() {
        // 读取data中对应属性值返回(实现代理读操作)
        return me._data[key];
      },
      // 当通过vm.name = 'xxx'时自动调用
      set: function proxySetter(newVal) {
        // 将最新的值保存到data中对应的属性上(实现代理写操作)
        me._data[key] = newVal;
      }
    });
  }

注解:方法会直接在一个对象上定义一个新属性,或者修改一个对象的现有属性, 并返回这个对象。如果不指定configurable, writable, enumerable ,则这些属性默认值为false,如果不指定value, get, set,则这些属性默认值为undefined。


模板解析(compile.js)

1.模板解析的关键对象: compile对象
2.模板解析的基本流程:
	1). 将el的所有子节点取出, 添加到一个新建的文档fragment对象中
	2). 对fragment中的所有层次子节点递归进行编译解析处理
    	* 对表达式文本节点进行解析
    	* 对元素节点的指令属性进行解析
        	* 事件指令解析
        	* 一般指令解析
  	3). 将解析后的fragment添加到el中显示
3.解析表达式文本节点: textNode.textContent = value
  	1). 根据正则对象得到匹配出的表达式字符串: 子匹配/RegExp.$1
  	2). 从data中取出表达式对应的属性值
  	3). 将属性值设置为文本节点的textContent
4.事件指令解析: elementNode.addEventListener(事件名, 回调函数.bind(vm))
    v-on:click="test"
  	1). 从指令名中取出事件名
  	2). 根据指令的值(表达式)从methods中得到对应的事件处理函数对象
  	3). 给当前元素节点绑定指定事件名和回调函数的dom事件监听
  	4). 指令解析完后, 移除此指令属性
5.一般指令解析: elementNode.xxx = value
  	1). 得到指令名和指令值(表达式)
  	2). 从data中根据表达式得到对应的值
  	3). 根据指令名确定需要操作元素节点的什么属性
        * v-text---textContent属性
        * v-html---innerHTML属性
        * v-class--className属性
  	4). 将得到的表达式的值设置到对应的属性上
  	5). 移除元素的指令属性
  • 主要步骤和代码
function Compile(el, vm) {
  // 保存
    this.$vm = vm;
  //保存el元素
    this.$ = this.isElementNode(el)?elj:document.querySelector(el);
  //如果el元素存在
    if(this.$el){
        //1. 取出el中所有子节点,封装在一个framgment中
        this.$framgment = this.node2Framgment(this.$el);
        //2. 编译framgment中所有层次子节点
        this.init();
        //3. 将framgment添加到el中
        this.#el.appendChild(this.$framgment);    
    }
}
  • init内部方法
compileElement: function (el) {
    // 得到所有子节点(el是this.$framgment)
    var childNodes = el.childNodes,
        // 保存compile对象
        me = this;
    // 遍历所有子节点
    [].slice.call(childNodes).forEach(function (node) {
        // 得到节点的文本内容
        var text = node.textContent;
        // 正则对象(匹配大括号表达式)
        var reg = /\{\{(.*)\}\}/;  // {{name}}
        // 如果是元素节点
        if (me.isElementNode(node)) {
            // 编译元素节点的指令属性
            me.compile(node);
            // 如果是一个大括号表达式格式的文本节点
        } else if (me.isTextNode(node) && reg.test(text)) {
            // 编译大括号表达式格式的文本节点
            me.compileText(node, RegExp.$1); // RegExp.$1: 表达式   name
        }
        // 如果子节点还有子节点
        if (node.childNodes && node.childNodes.length) {
            // 递归调用实现所有层次节点的编译
            me.compileElement(node);
        }
    });
}
,

compile: function (node) {
    // 得到所有标签属性节点
    var nodeAttrs = node.attributes,
        me = this;
    // 遍历所有属性
    [].slice.call(nodeAttrs).forEach(function (attr) {
        // 得到属性名: v-on:click
        var attrName = attr.name;
        // 判断是否是指令属性
        if (me.isDirective(attrName)) {
            // 得到表达式(属性值): test
            var exp = attr.value;
            // 得到指令名: on:click
            var dir = attrName.substring(2);
            // 事件指令
            if (me.isEventDirective(dir)) {
                // 解析事件指令
                compileUtil.eventHandler(node, me.$vm, exp, dir);
                // 普通指令
            } else {
                // 解析普通指令
                compileUtil[dir] && compileUtil[dir](node, me.$vm, exp);
            }

            // 移除指令属性
            node.removeAttribute(attrName);
        }
    });
}
,
compileText: function (node, exp) {
    // 调用编译工具对象解析
    compileUtil.text(node, this.$vm, exp);
}
,

// 指令处理集合
var compileUtil = {
    // 解析: v-text/{{}}
    text: function (node, vm, exp) {
        this.bind(node, vm, exp, 'text');
    },
    // 解析: v-html
    html: function (node, vm, exp) {
        this.bind(node, vm, exp, 'html');
    },

    // 解析: v-model
    model: function (node, vm, exp) {
        this.bind(node, vm, exp, 'model');

        var me = this,
            val = this._getVMVal(vm, exp);
        node.addEventListener('input', function (e) {
            var newValue = e.target.value;
            if (val === newValue) {
                return;
            }

            me._setVMVal(vm, exp, newValue);
            val = newValue;
        });
    },

    // 解析: v-class
    class: function (node, vm, exp) {
        this.bind(node, vm, exp, 'class');
    },

    // 真正用于解析指令的方法
    bind: function (node, vm, exp, dir) {
        /*实现初始化显示*/
        // 根据指令名(text)得到对应的更新节点函数
        var updaterFn = updater[dir + 'Updater'];
        // 如果存在调用来更新节点
        updaterFn && updaterFn(node, this._getVMVal(vm, exp));

        // 创建表达式对应的watcher对象
        new Watcher(vm, exp, function (value, oldValue) {/*更新界面*/
            // 当对应的属性值发生了变化时, 自动调用, 更新对应的节点
            updaterFn && updaterFn(node, value, oldValue);
        });
    },

    // 事件处理
    eventHandler: function (node, vm, exp, dir) {
        // 得到事件名/类型: click
        var eventType = dir.split(':')[1],
            // 根据表达式得到事件处理函数(从methods中): test(){}
            fn = vm.$options.methods && vm.$options.methods[exp];
        // 如果都存在
        if (eventType && fn) {
            // 绑定指定事件名和回调函数的DOM事件监听, 将回调函数中的this强制绑定为vm
            node.addEventListener(eventType, fn.bind(vm), false);
        }
    },

    // 得到表达式对应的value
    _getVMVal: function (vm, exp) {
        var val = vm._data;
        exp = exp.split('.');
        exp.forEach(function (k) {
            val = val[k];
        });
        return val;
    },

    _setVMVal: function (vm, exp, value) {
        var val = vm._data;
        exp = exp.split('.');
        exp.forEach(function (k, i) {
            // 非最后一个key,更新val的值
            if (i < exp.length - 1) {
                val = val[k];
            } else {
                val[k] = value;
            }
        });
    }
};

// 包含多个用于更新节点方法的对象
var updater = {
    // 更新节点的textContent
    textUpdater: function (node, value) {
        node.textContent = typeof value == 'undefined' ? '' : value;
    },

    // 更新节点的innerHTML
    htmlUpdater: function (node, value) {
        node.innerHTML = typeof value == 'undefined' ? '' : value;
    },

    // 更新节点的className
    classUpdater: function (node, value, oldValue) {
        var className = node.className;
        className = className.replace(oldValue, '').replace(/\s$/, '');

        var space = className && String(value) ? ' ' : '';

        node.className = className + space + value;
    },

    // 更新节点的value
    modelUpdater: function (node, value, oldValue) {
        node.value = typeof value == 'undefined' ? '' : value;
    }
};

数据劫持–>数据绑定

1.数据绑定(model==>View):

  • 一旦更新了data中的某个属性数据, 所有界面上直接使用或间接使用了此属性的节点都会更新(更新)

2.数据劫持

  • 数据劫持是vue中用来实现数据绑定的一种技术
  • 基本思想: 通过defineProperty()来监视data中所有属性(任意层次)数据的变化, 一旦变化就去更新界面

3.四个重要对象

1.Observer

  • 用来对data所有属性数据进行劫持的构造函数
  • 给data中所有属性重新定义属性描述(get/set)
  • 为data中的每个属性创建对应的dep对象

2. Dep(Depend)

  • data中的每个属性(所有层次)都对应一个dep对象
  • 创建的时机:
    * 在初始化define data中各个属性时创建对应的dep对象
    * 在data中的某个属性值被设置为新的对象时
  • 对象的结构
{
id, // 每个dep都有一个唯一的id
subs //包含n个对应watcher的数组(subscribes的简写)
}
  • subs属性说明:当一个watcher被创建时, 内部会将当前watcher对象添加到对应的dep对象的subs中,当此data属性的值发生改变时, 所有subs中的watcher都会收到更新的通知, 从而最终更新对应的界面

3. Compile

  • 用来解析模板页面的对象的构造函数(一个实例)
  • 利用compile对象解析模板页面
  • 每解析一个表达式(非事件指令)都会创建一个对应的watcher对象, 并建立watcher与dep的关系
  • complie与watcher关系: 一对多的关系

4. Watcher

  • 模板中每个非事件指令或表达式都对应一个watcher对象
    • 监视当前表达式数据的变化
    • 创建的时机: 在初始化编译模板时
    • 对象的组成
{
    vm,  //vm对象
    exp, //对应指令的表达式
    cb, //当表达式所对应的数据发生改变的回调函数
    value, //表达式当前的值
    depIds //表达式中各级属性所对应的dep对象的集合对象属性名为dep的id, 属性值为dep
}

5. 总结: dep与watcher的关系: 多对多

  • 一个data中的属性对应对应一个dep, 一个dep中可能包含多个watcher(模板中有几个表达式使用到了属性)
  • 模板中一个非事件表达式对应一个watcher, 一个watcher中可能包含多个dep(表达式中包含了几个data属性)
  • 数据绑定使用到2个核心技术
    • defineProperty()
    • 消息订阅与发布

4. 双向数据绑定

  • 双向数据绑定是建立在单向数据绑定(model==>View)的基础之上的
  • 双向数据绑定的实现流程:
    1. 在解析v-model指令时, 给当前元素添加input监听
    2. 当input的value发生改变时, 将最新的值赋值给当前表达式所对应的data属性

MVVM原理图

Vue源码学习记录_第1张图片
流程:

  • 初始化:新建mvvm模型,创建Observer监听data中所有层次的属性,与data中的属性dep一一对应。同时创建了Compile解析指令属性,调用updater初始化视图,并为视图创建对应的watcher,指定更新的函数。在创建Watcher时,为dep添加订阅者wathcer。
  • 更新: js对象属性发生变化,Observer监听到属性变化,通知dep。dep通过消息订阅方式通知watcher,watcher调用updater更新界面视图。

你可能感兴趣的:(前端)