您的位置:68399皇家赌场 > 服务器租用 > Vue原理深入分析 完结双向绑定MVVM

Vue原理深入分析 完结双向绑定MVVM

发布时间:2019-06-07 19:55编辑:服务器租用浏览(109)

    基本原理

    MVC模式

    MVC模式

    早年的MVC格局是单向绑定,即Model绑定到View,当大家用JavaScript代码更新Model时,View就能够自动更新

    正文能帮您做怎么着?

    正文能帮你做哪些?
    一、精晓vue的双向数据绑定原理以及基本代码模块
    2、缓和好奇心的还要了然哪些兑现双向绑定
    为了便于表达原理与完毕,本文相关代码主要摘自vue源码, 并进行了简化改变,相对较简陋,并没有怀想到数组的管理、数据的轮回放重等,也未免存在一些难点,迎接我们指正。可是那些并不会潜移默化我们的阅读和透亮,相信看完本文后对我们在翻阅vue源码的时候会更有帮扶<
    正文全体有关代码均在github上边可找到
    信任大家对mvvm双向绑定应该都不目生了,一言不合上代码,上边先看三个本文末了落到实处的功效啊,和vue一样的语法。

    Vue.采纳数据威迫结合发布者-订阅者方式的法门,通过Object.defineProperty()来要挟各样属性的setter和getter,数据变动时发布音信给订阅者,触发相应函数的回调。

    MVVM模式

    MVVM形式便是Model–View–ViewModel情势。它完结了View的改观,自动反映在 ViewModel,反之亦然。
    自小编对于双向绑定的接头,正是用户更新了View,Model的数据也自行被更新了,这种气象正是双向绑定。再说细点,就是在另一方面绑定的根基上给可输入成分(input、textare等)增加了change(input)事件,(change事件触发,View的情事就被更新了)来动态修改model。

    MVVM模式

    一、理解vue的双向数据绑定原理以及着力代码模块
    2、减轻好奇心的同时询问怎么落实双向绑定
    为了便于表达原理与落到实处,本文相关代码主要摘自vue源码, 并进行了简化改换,相对较简陋,并未有思虑到数组的管理、数据的大循环正视等,也免不了存在有的标题,应接大家指正。可是那么些并不会影响我们的阅读和了然,相信看完本文后对大家在阅读vue源码的时候会更有帮带<
    正文全部有关代码均在github上边可找到
    相信大家对mvvm双向绑定应该都不目生了,一言不合上代码,下边先看三个本文最后促成的机能啊,和vue同样的语法,假设还不打听双向绑定,猛戳谷歌(Google)

    <div id="mvvm-app">
     <input type="text" v-model="word">
     <p>{{word}}</p>
     <button v-on:click="sayHi">change model</button>
    </div>
    
    <script src="./js/observer.js"></script>
    <script src="./js/watcher.js"></script>
    <script src="./js/compile.js"></script>
    <script src="./js/mvvm.js"></script>
    <script>
     var vm = new MVVM({
      el: '#mvvm-app',
      data: {
       word: 'Hello World!'
      },
      methods: {
       sayHi: function() {
        this.word = 'Hi, everybody!';
       }
      }
     });
    </script>
    

    思路整理

    双向绑定原理

    vue数据双向绑定是由此数据威迫结合发表者-订阅者情势的章程来达成的。
    我们曾经知道完成多少的双向绑定,首先要对数码举办胁制监听,所以大家须要设置八个监听器Observer,用来监听全部属性。如果属性发上扭转了,就供给报告订阅者沃特cher看是不是需求创新。因为订阅者是有无数个,所以大家必要有三个音讯订阅器Dep来特地采访这个订阅者,然后在监听器Observer和订阅者沃特cher之间进行联合保管的。接着,大家还需求有二个指令分析器Compile,对各样节点成分实行围观和解析,将有关指令(如v-model,v-on)对应开始化成八个订阅者沃特cher,并替换模板数据也许绑定相应的函数,此时当订阅者沃特cher接收到对应属性的生成,就能够实践相应的立异函数,从而创新视图。由此接下去我们推行以下三个步骤,完成多少的双向绑定:

    一.贯彻2个监听器Observer,用来威胁并监听全部属性,要是有变动的,就通报订阅者。

    二.落实二个订阅者沃特cher,每3个沃特cher都绑定二个翻新函数,watcher能够接到属性的成形文告并实践相应的函数,从而立异视图。

    3.贯彻2个剖判器Compile,能够扫描和剖判每种节点的连带指令(v-model,v-on等一声令下),假设节点存在v-model,v-on等一声令下,则深入分析器Compile初阶化那类节点的沙盘数据,使之能够来得在视图上,然后开端化相应的订阅者(沃特cher)。

    <div id="mvvm-app">
     <input type="text" v-model="word">
     <p>{{word}}</p>
     <button v-on:click="sayHi">change model</button>
    </div>
    
    <script src="./js/observer.js"></script>
    <script src="./js/watcher.js"></script>
    <script src="./js/compile.js"></script>
    <script src="./js/mvvm.js"></script>
    <script>
     var vm = new MVVM({
     el: '#mvvm-app',
     data: {
      word: 'Hello World!'
     },
     methods: {
      sayHi: function() {
      this.word = 'Hi, everybody!';
      }
     }
     });
    </script>
    

    效果:

    要完毕mvvm的双向绑定,须要达成如下几点:

    壹.实现四个Observer

    Observer是三个数量监听器,其得以实现宗旨措施正是Object.defineProperty( )。借使要对具备属性都开展监听的话,那么能够因此递归方法遍历全体属性值,并对其张开Object.defineProperty( )管理
    如下代码完成了1个Observer。

    function Observer(data) {
        this.data = data;
        this.walk(data);
    }
    
    Observer.prototype = {
        walk: function(data) {
            var self = this;
            //这里是通过对一个对象进行遍历,对这个对象的所有属性都进行监听
            Object.keys(data).forEach(function(key) {
                self.defineReactive(data, key, data[key]);
            });
        },
        defineReactive: function(data, key, val) {
            var dep = new Dep();
          // 递归遍历所有子属性
            var childObj = observe(val);
            Object.defineProperty(data, key, {
                enumerable: true,
                configurable: true,
                get: function getter () {
                    if (Dep.target) {
                      // 在这里添加一个订阅者
                      console.log(Dep.target)
                        dep.addSub(Dep.target);
                    }
                    return val;
                },
               // setter,如果对一个对象属性值改变,就会触发setter中的dep.notify(),通知watcher(订阅者)数据变更,执行对应订阅者的更新函数,来更新视图。
                set: function setter (newVal) {
                    if (newVal === val) {
                        return;
                    }
                    val = newVal;
                  // 新的值是object的话,进行监听
                    childObj = observe(newVal);
                    dep.notify();
                }
            });
        }
    };
    
    function observe(value, vm) {
        if (!value || typeof value !== 'object') {
            return;
        }
        return new Observer(value);
    };
    
    // 消息订阅器Dep,订阅器Dep主要负责收集订阅者,然后在属性变化的时候执行对应订阅者的更新函数
    function Dep () {
        this.subs = [];
    }
    Dep.prototype = {
      /**
       * [订阅器添加订阅者]
       * @param  {[Watcher]} sub [订阅者]
       */
        addSub: function(sub) {
            this.subs.push(sub);
        },
      // 通知订阅者数据变更
        notify: function() {
            this.subs.forEach(function(sub) {
                sub.update();
            });
        }
    };
    Dep.target = null;
    

    在Observer中,当初笔者看人家的源码时,小编有有个别不领会的地点正是Dep.target是从哪个地方来的,相信有些人和作者会有一致的疑云。这里不急急,当写到沃特cher的时候,你就能够发觉,这么些Dep.target是根源Watcher。

    效果:

    www.68399.com 1

    壹.落实1个数目监听器Observer,能够对目的的有所属性进行监听,产生变化时得到新型值布告订阅者
    二.已毕1个深入分析器Compile,对种种子成分节点的授命张开扫描和深入分析,依据模板指令替换数据,起始化视图以及绑定相应的回调函数;
    3.贯彻1个沃特cher,作为Observer和Compile的大桥,能够订阅属性别变化动的布告,实施命令绑定的回调函数,更新视图
    四.mvvm的进口,整合上述三者

    二.兑现3个沃特cher

    Watcher正是1个订阅者。用于将Observer发来的update新闻处理,试行沃特cher绑定的翻新函数。
    一般来讲代码完毕了三个沃特cher

    function Watcher(vm, exp, cb) {
        this.cb = cb;
        this.vm = vm;
        this.exp = exp;
        this.value = this.get();  // 将自己添加到订阅器的操作
    }
    
    Watcher.prototype = {
        update: function() {
            this.run();
        },
        run: function() {
            var value = this.vm.data[this.exp];
            var oldVal = this.value;
            if (value !== oldVal) {
                this.value = value;
                this.cb.call(this.vm, value, oldVal);
            }
        },
        get: function() {
            Dep.target = this;  // 缓存自己
            var value = this.vm.data[this.exp]  // 强制执行监听器里的get函数
            Dep.target = null;  // 释放自己
            return value;
        }
    };
    

    在本身研商代码的进程中,笔者以为最复杂的正是清楚那些函数的参数,后来在自己出口了那一个参数之后,函数的这个效应也轻易领会了。vm,正是事后要写的SelfValue对象,相当于Vue中的new Vue的三个对象。exp是node节点的v-model或v-on:click等一声令下的属性值。如v-model="name",exp就是"name"。cb,便是沃特cher绑定的更新函数。
    下边包车型地铁代码中就能够看出来,在沃特cher的getter函数中,Dep.target指向了温馨,也等于沃特cher对象。在getter函数中,

    var value = this.vm.data[this.exp]  // 强制执行监听器里的get函数。
    

    这里得到vm.data[this.exp] 时,会调用Observer中Object.defineProperty中的get函数

    get: function getter () {
                    if (Dep.target) {
                      // 在这里添加一个订阅者
                      console.log(Dep.target)
                        dep.addSub(Dep.target);
                    }
                    return val;
                },
    

    之所以把watcher增添到了订阅器中,也就缓慢解决了下边Dep.target是何地来的那么些难点。

    www.68399.com 2

    二种实现双向绑定的做法

    流程图如下:

    3.贯彻叁个Compile

    new SelfVue 绑定的dom节点

    Compile主要的魔法是把new SelfVue 绑定的dom节点,(也正是el标签绑定的id)遍历该节点的全数子节点,找寻当中全部的v-指令和" {{}} ".
    一.假设实节点含有v-指令,就是成分节点,则对那么些因素增加监听事件。(若是是v-on,则node.add伊芙ntListener('click'),假使是v-model,则node.add伊芙ntListener('input'))。接着早先化模板成分,创制三个沃特cher绑定这几个因三秋点。
    二.假设子节点是文本节点,即" {{ data }} ",则用正则表达式抽取" {{ data }} "中的data,然后var initText = this.vm[exp],用initText去顶替个中的data。
    实际代码参见小编的github: vue-MVVM
    里头有详实的评释。

    二种完毕双向绑定的做法

    近期两种主流的mvc(vm)框架都落到实处了单向数据绑定,而小编所精晓的双向数据绑定无非正是在单方面绑定的底蕴上给可输入成分(input、textare等)增添了change(input)事件,来动态修改model和 view,并从未多高深。所以无需太过介怀是实现的单向或双向绑定。

    www.68399.com 3

    四.兑现叁个MVVM

    可以说MVVM是Observer,Compile以及沃特cher的“boss”了,他索要配备给Observer,Compile以及沃特che做的事务如下

    a、Observer完结对MVVM自个儿model数据威迫,监听数据的性格退换,并在改变时开始展览notify
    b、Compile完结指令分析,开首化视图,并订阅数据变化,绑定好更新函数
    c、沃特cher壹方面接收Observer通过dep传递过来的数目变化,一方面布告Compile进行view update。
    终极,把那么些MVVM抽象出来,便是vue中Vue的构造函数了,可以组织出2个vue实例。

    近日两种主流的mvc(vm)框架都落到实处了单向数据绑定,而自己所知道的双向数据绑定无非便是在一面绑定的基础上给可输入成分(input、textare等)加多了change(input)事件,来动态修改model和 view,并从未多高深。所以没有必要太过介怀是兑现的单向或双向绑定。

    福寿双全数量绑定的做法有大概如下两种:

    布满实现

    最后写叁个html测试一下大家的法力

    <!DOCTYPE html>
    <html lang="en">
    <head>
        <meta charset="UTF-8">
        <title>self-vue</title>
    </head>
    <style>
        #app {
            text-align: center;
        }
    </style>
    <body>
        <div id="app">
            <h2>{{title}}</h2>
            <input v-model="name">
            <h1>{{name}}</h1>
            <button v-on:click="clickMe">click me!</button>
        </div>
    </body>
    <script src="js/observer.js"></script>
    <script src="js/watcher.js"></script>
    <script src="js/compile.js"></script>
    <script src="js/mvvm.js"></script>
    <script type="text/javascript">
    
         var app = new SelfVue({
            el: '#app',
            data: {
                title: 'hello world',
                name: 'canfoo'
            },
            methods: {
                clickMe: function () {
                    this.title = 'hello world';
                }
            },
            mounted: function () {
                window.setTimeout(() => {
                    this.title = '你好';
                }, 1000);
            }
        });
    
    </script>
    </html>
    

    先执行mvvm中的new SelfVue(...),在mvvm.js中,

    observe(this.data);
    new Compile(options.el, this);
    

    先起头化四个监听器Observer,用于监听该指标data属性的值。
    下一场早先化七个深入分析器Compile,绑定那一个节点,并剖析在那之中的v-," {{}} "指令,(每多少个发令对应二个沃特cher)并早先化模板数据以及开始化相应的订阅者,并把订阅者增加到订阅器中(Dep)。那样就兑现双向绑定了。
    只要v-model绑定的要素,

    <input v-model="name">
    

    即输入框的值爆发变化,就能够触发Compile中的

    node.addEventListener('input', function(e) {
                var newValue = e.target.value;
                if (val === newValue) {
                    return;
                }
                self.vm[exp] = newValue;
                val = newValue;
            });
    

    self.vm[exp]www.68399.com, = newValue;这么些语句会触发mvvm中SelfValue的setter,以及触发Observer对该目的name属性的监听,即Observer中的Object.defineProperty()中的setter。setter中有文告订阅者的函数dep.notify,沃特cher收到布告后就能奉行绑定的创新函数。
    最后的末梢正是功能图啦:

    双向绑定

    相关参照他事他说加以考查链接:http://www.cnblogs.com/canfoo/p/6891868.html

    兑现多少绑定的做法有大要如下两种:

    公布者-订阅者方式(backbone.js)

    1. MVVM.js

    • 发表者-订阅者情势(backbone.js)
    • 脏值检查(angular.js)
    • 多少勒迫(vue.js)

    脏值检查(angular.js)

    function MVVM(options) {
      this.$options = options || {};
      var data = this._data = this.$options.data;
      var me = this;
    
      // 数据代理
      // 实现 vm.xxx -> vm._data.xxx
      Object.keys(data).forEach(function(key) {
        me._proxyData(key);
      });
      // 代理计算属性
      // 同样通过Object.defineProperty进行劫持
      this._initComputed();
    
      observe(data, this);
    
      this.$compile = new Compile(options.el || document.body, this)
    }
    
    MVVM.prototype = {
      $watch: function(key, cb, options) {
        new Watcher(this, key, cb);
      }
    }
    

    发布者-订阅者格局: 一般通过sub, pub的点子贯彻多少和视图的绑定监听,更新数据格局一般做法是 vm.set('property', value),这里有篇小说讲的可比详细,风乐趣可点这里

    多少威迫(vue.js)

    MVVM入口文件,整合Observer/Compile/沃特cher3者,达到多少变化->更新视图;视图变化->数据变动的双向绑定效果。(结合钩子函数,精晓Vue生命周期中逐壹阶段的效果)

    这种措施现在到底太low了,大家更期待通过 vm.property = value 这种艺术更新数据,同一时候自动更新视图,于是有了上面两种方法

    发表者-订阅者格局: 一般通过sub, pub的主意贯彻多少和视图的绑定监听,更新数据模式一般做法是 vm.set('property', value) ,这里有篇小说讲的可比详细,风乐趣可点这里

    2. Observer.js

    脏值检查: angular.js 是通过脏值检查测试的办法比对数据是还是不是有转移,来支配是还是不是更新视图,最简便易行的点子正是由此setInterval() 定时轮询检验数据变动,当然谷歌不会那样low,angular唯有在钦命的风云触发时进入脏值检查评定,大约如下:

    这种办法今后到底太low了,大家更期待通过 vm.property = value 这种措施创新数据,相同的时间自动更新视图,于是有了下边三种方法

    function Observer(data) {
      Object.keys(data).forEach(function() {
        defineReactive(data, key, data[key]);
      });
    }
    function defineReactive (data, key, val) {
      var dep = new Dep();
      var childObj = observe(val);
    
      Object.defineProperty(data, key, {
        enumerable: true, // 可枚举
        configurable: false, // 不能再define
        get: function() {
          if (Dep.target) {
            dep.depend();
          }
          return val;
        },
        set: function(newVal) {
          if (newVal === val) {
            return;
          }
          val = newVal;
          // 新的值是object的话,进行监听
          childObj = observe(newVal);
          // 通知订阅者
          dep.notify();
        }
      });
    }
    
    • DOM事件,比方用户输入文本,点击开关等。( ng-click )
    • XHR响应事件 ( $http )
    • 浏览器Location退换事件 ( $location )
    • Timer事件( $timeout , $interval )
    • 执行 $digest() 或 $apply()

    脏值检查: angular.js 是通过脏值检验的点子比对数据是不是有改动,来支配是不是更新视图,最简便易行的秘技正是透过 setInterval() 定期轮询检测数据变动,当然谷歌(Google)不会如此low,angular唯有在钦点的事件触发时进入脏值检查测试,大约如下:

    对亟待监测的靶子的每一个属性举办递归遍历,通过Object.defineProperty设置setter和getter。当设置新的属性值时,触发相应的setter,通告订阅者。

    数据勒迫: vue.js 则是选择数据恐吓结合揭橥者-订阅者形式的艺术,通过Object.defineProperty()来胁制各样属性的setter,getter,在数据变动时发布音讯给订阅者,触发相应的监听回调。

    • DOM事件,例如用户输入文本,点击按键等。( ng-click )
    • XH安德拉响应事件 ( $http )
    • 浏览器Location改换事件 ( $location )
    • Timer事件( $timeout , $interval )
    • 执行 $digest() 或 $apply()

    本文由68399皇家赌场发布于服务器租用,转载请注明出处:Vue原理深入分析 完结双向绑定MVVM

    关键词: 68399皇家赌场 日记本 Vue集结号-...

上一篇:基于layui数据表格以及传数据的法子

下一篇:没有了