vue数据绑定原理

关于数据绑定

在前端开发中使用MV*模型的时候,M—model,指的是模型,即指的是数据,V—view指的是视图,就是页面展示的部分。
通常所说的数据的双向绑定就是

关于框架中常用的数据绑定做法

  • 发布者-订阅者模式(backcone.js)
  • 脏值检测(angular.js)
  • 数据劫持(vue.js)

思路整理

vue使用数据劫持+发布者-订阅者模式进行双向数据绑定,其中最核心的方法就是通过Object.defineProperty()实现数据的劫持,达到监听数据的变动的目的。
整理一下vue进行数据绑定的思路:

  1. 实现一个数据监听器Observer,能够进行数据对象的属性监听和更新。
  2. 实现一个指令解析器Compile,对每个元素节点的指令进行扫描和解析,并且根据指令进行绑定值的提取。
  3. 实现一个Watcher作为连接Observer和Compile的桥梁,并可以接受到属性的变动,进行相应属性的更新,从而实现视图的更新。
  4. 使用Vue入口函数,整合三者。

实现双向绑定

  • 入口函数Vue编写
function Vue(options) {//options就是new Vue传递对象参数
        this.data = options.data;
        this.id = options.el;
        Observer(this.data, this);//进行数据劫持和更新
        var app = document.getElementById(this.id);//div
        var dom = nodeToFragment(app, this);//进行虚拟DOM操作
        app.appendChild(dom);//将虚拟DOM插入文档中
    }
  • 数据劫持函数Observer
  function Observer(data, vm) {
        Object.keys(data).forEach(function (key) {//循环遍历出每一个值进行数据劫持
            defineReactive(vm, key, data[key]);
        })
    }

    function defineReactive(vm, key, val) {
        Object.defineProperty(vm, key, {//当数据发生变化时调用set进行更新,获取数据时使用set
            get: function () {
                return val;
            },
            set: function (newVal) {
                if (newVal === val) {
                    return;
                }
                val = newVal;
            }
        })
        return val;
    }
  • 指令解析函数Compile
 function nodeToFragment(node, vm) {
        var flag = document.createDocumentFragment();
        var child;
        while (child = node.firstChild) {
            Compile(child, vm);
            flag.appendChild(child);
        }
        return flag;
    }

function Compile(node, vm) {
        var reg = /\{\{(.*)\}\}/;
        if (node.nodeType === 1) {
            var attr = node.attributes;
            for (var i = 0; i < attr.length; i++) {
                if (attr[i].nodeName === 'v-model') {
                    var name = attr[i].nodeValue;
                    node.value = vm[name];
                    node.removeAttribute('v-model');
                }
            }
        }
        if (node.nodeType === 3) {
            if (reg.test(node.nodeValue)) {
                var name = RegExp.$1;
                node.nodeValue = vm[name];
            }
        }
    }
  • node.nodeType
  1. 元素节点 =1
  2. 属性节点=2
  3. 文本节点= 3
  4. 注释节点= 8
  • node.attributes属性
用法:
document.getElementsById("app").attributes;
node.attributes

描述
attributes 属性返回指定节点的属性集合,即 NamedNodeMap。
提示:您可以使用 length 属性来确定属性的数量,然后您就能够遍历所有的属性节点并提取您需要的信息。

  • 订阅者-发布者模式Watcher
    Watcher订阅者作为Observer和Compile之间的桥梁,主要做一下工作:
  1. 编写属性订阅器Dep,可以调用notice进行Watcher的更新,和添加订阅者
  2. 自身有一个update()方法
  3. 当属性调用notice()方法时可以进行调用update()方法
  function Dep() {
        this.subs = [];
    }

    Dep.prototype = {
        addSub: function (sub) {
            this.subs.push(sub);//增加订阅者
        },
        notify: function () {
            this.subs.forEach(function (sub) {
                sub.update();
            })
        }
    }
function watcher(vm, node, name, nodeType) {
        Dep.target = this;
        this.vm = vm;
        this.node = node;
        this.name = name;
        this.nodeType = nodeType;
        this.update();
        Dep.target = null;
    }

    watcher.prototype = {
        update: function () {
            this.get();
            if (this.nodeType == 'text') {
                this.node.nodeValue = this.value;
            }
            if (this.nodeType == 'input') {
                this.node.value = this.value;
            }
        },
        //获取data中属性
        get: function () {
            this.value = this.vm[this.name];
        }
    }
  • 最后
  1. 进行Vue的声明
 var App = new Vue({
        el: 'app',
        data: {
            title: 'hello'
        }
    })
  1. 在Compile中进行事件触发,以及watcher的调用
 node.addEventListener('input', function (e) {
       vm[name] = e.target.value;
  })
 new Watcher(vm, node, name);
  1. 数据劫持时添加dep
function defineReactive(vm, key, val) {
        var dep = new Dep();
        Object.defineProperty(vm, key, {
            get: function () {
                if (Dep.target) {
                    dep.addSub(Dep.target);
                }
                return val;
            },
            set: function (newVal) {
                if (newVal === val) {
                    return;
                }
                val = newVal;
                dep.notify();
            }
        })
        return val;
    }
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • 本篇文章中的代码只是部分片段,完整代码存放于github上https://github.com/Q-Zhan/si...
    詹前鑫阅读 2,140评论 0 5
  • 我的github: vue双向绑定原理 MVC模式 以往的MVC模式是单向绑定,即Model绑定到View,当我们...
    KlausXu阅读 45,424评论 7 90
  • vue理解浅谈 一 理解vue的核心理念 使用vue会让人感到身心愉悦,它同时具备angular和react的优点...
    ndxs2008阅读 24,409评论 2 18
  • 这方面的文章很多,但是我感觉很多写的比较抽象,本文会通过举例更详细的解释。(此文面向的Vue新手们,如果你是个大牛...
    Ivy_2016阅读 15,584评论 8 63
  • 颊骨下出现凹陷——皮肤松弛的前兆。 眼角有细纹——干纹和鱼尾纹的前身。 嘴唇非常干燥、细纹增多、缺少润泽感——唇部...
    塞瓶器阅读 134评论 0 1

友情链接更多精彩内容