当前位置: 移动技术网 > IT编程>开发语言>JavaScript > vue中的双向数据绑定原理与常见操作技巧详解

vue中的双向数据绑定原理与常见操作技巧详解

2020年05月10日  | 移动技术网IT编程  | 我要评论

本文实例讲述了vue中的双向数据绑定原理与常见操作技巧。分享给大家供大家参考,具体如下:

什么是双向数据绑定?

vue是一个mvvm框架,即数据双向绑定,即当数据发生变化的时候,视图也就发生变化,当视图发生变化的时候,数据也会跟着同步变化。这也是算是vue的精髓之处了。值得注意的是,我们所说的数据双向绑定,一定是对于ui控件来说的,非ui控件不会涉及到数据双向绑定。单向数据绑定是使用状态管理工具的前提,如果我们使用vuex,那么数据流也是单向的,这时就会和双向数据绑定有冲突,我们可以这么解决。

为什么要实现数据的双向绑定?

在vue中,如果使用vuex,实际上数据还是单向的,之所以说是数据双向绑定,这是用的ui控件来说,对于我们处理表单,vue的双向数据绑定用起来就特别舒服了。即两者并不互斥,在全局性数据流使用单项,方便跟踪,局部性数据流使用双向,简单易操作。

1.访问器属性

object.defineproperty()函数可以定义对象的属性相关描述符,其中的set和get函数对于完成数据双向绑定起到了至关重要的作用,下面,我们看看这个函数的基本使用方式。

var obj = {
   foo: 'foo'
  }

  object.defineproperty(obj, 'foo', {
   get: function () {
    console.log('将要读取obj.foo属性');
   }, 
   set: function (newval) {
    console.log('当前值为', newval);
   }
  });

  obj.foo; // 将要读取obj.foo属性
  obj.foo = 'name'; // 当前值为 name

上面代码中,get即为我们访问属性时调用,set为我们设置属性值时调用。

2.简单的数据双向绑定实现方法

<!doctype html>
<html lang="en">
<head>
 <meta charset="utf-8">
 <title>forvue</title>
</head>
<body>
 <input type="text" id="textinput">
 输入:<span id="textspan"></span>
 <script>
  var obj = {},
    textinput = document.queryselector('#textinput'),
    textspan = document.queryselector('#textspan');

  object.defineproperty(obj, 'foo', {
   set: function (newvalue) {
    textinput.value = newvalue;
    textspan.innerhtml = newvalue;
   }
  });

  textinput.addeventlistener('keyup', function (e) {
    obj.foo = e.target.value;
  });

 </script>
</body>
</html>

可以看到,实现一个简单的数据双向绑定还是不难的,使用object.defineproperty()来定义属性的set函数,属性被赋值的时候,修改input的value值以及span中的innerhtml,然后监听input的keyup事件,修改对象的属性值,即可以实现这样一个简单的数据双向绑定。

3. 实现任务的思路

上面我们只是实现了一个简单的数据双向绑定,而我们真正希望实现的是下面这种方式:

<div id="app">
    <input type="text" v-model="text">
    {{ text }}
  </div> 

  <script>
    var vm = new vue({
      el: '#app', 
      data: {
        text: 'hello world'
      }
    });
  </script>

即和vue一样的方式来实现数据的双向绑定,那么我们可以把整个实现过程分为下面几步:

输入框以及文本节点与data中的数据绑定

输入框内容变化时,data中的数据同步变化。即view => model的变化。

data中的数据变化 时,文本节点的内容同步变化。即model => view的变化。

4.documentfragment

如果希望实现任务,我们还需要使用到documentfragment文档片段,可以把它看做一个容器,如下所示:

<div id="app">
    
  </div>
  <script>
    var flag = document.createdocumentfragment(),
      span = document.createelement('span'),
      textnode = document.createtextnode('hello world');
    span.appendchild(textnode);
    flag.appendchild(span);
    document.queryselector('#app').appendchild(flag)
  </script>

使用文档片段的好处在于:在文档片段上进行操作dom,而不会影响到真实的dom,操作完成后,我们就可以添加到真实的dom上,这样的效率比直接在正式dom上修改要高很多。

vue在进行编译时,就是将挂载目标的所有子节点劫持到documentfragment中,经过一番处理之后,再将documentfragment整体返回插入挂载目标。

5.初始化数据绑定

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.data[name]
    node.removeattribute('v-model')
   }
   
  }
 }
 
 if (node.nodetype === 3) {
  if (reg.test(node.nodevalue)) {
   var name = regexp.$1
   name = name.trim()
   node.nodevalue = vm.data[name]
  }
 }
}

function nodetofragment(node, vm) {
 var flag = document.createdocumentfragment()
 var child 
 while(child = node.firstchild) {
  compile(child, vm)
  flag.appendchild(child)
 }
 return flag
}

function vue(options) {
 this.data = options.data 
 var el = options.el
 var dom = nodetofragment(document.queryselector(el), this)
 
 document.queryselector(el).appendchild(dom)
}

var vm = new vue({
 el: '#app',
 data: {
  text: 'hello'
 }
})

6.响应式的数据绑定

我们再来看看任务的实现思路,当我们在输入框输入数据的时候,首先触发input事件(或者keyup,change事件),在相应的事件处理程序中,我们获取输入框的value并赋值给vm实例的text属性。我们会利用defineproperty将data中text设置为vm的访问器属性,因此给vm.text赋值,就会触发set方法。在set方法可主要做两件事,第一,更新属性的值,第二后面再说。

<!doctype html>
<html lang="en">
<head>
 <meta charset="utf-8">
 <title>forvue</title>
</head>
<body>
  <div id="app">
    <input type="text" v-model="text">
    {{ text }}
  </div>
    
  <script>
    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; // 获取v-model绑定的属性名
            node.addeventlistener('input', function (e) {
              // 给相应的data属性赋值,进而触发属性的set方法
              vm[name] = e.target.value;
            })


            node.value = vm[name]; // 将data的值赋值给该node
            node.removeattribute('v-model');
          }
        }
      }

      // 节点类型为text
      if (node.nodetype === 3) {
        if (reg.test(node.nodevalue)) {
          var name = regexp.$1; // 获取匹配到的字符串
          name = name.trim();
          node.nodevalue = vm[name]; // 将data的值赋值给该node
        }
      }
    }

    function nodetofragment(node, vm) {
      var flag = document.createdocumentfragment();
      var child;

      while (child = node.firstchild) {
        compile(child, vm);
        flag.appendchild(child); // 将子节点劫持到文档片段中
      }
      
      return flag;
    }

    function vue(options) {
      this.data = options.data;
      var data = this.data;

      observe(data, this);

      var id = options.el;
      var dom = nodetofragment(document.getelementbyid(id), this);
      // 编译完成后,将dom返回到app中。
      document.getelementbyid(id).appendchild(dom);
    }

    var vm = new vue({
      el: 'app',
      data: {
        text: 'hello world'
      }
    });

    function definereactive(obj, key, val) {
      // 响应式的数据绑定
      object.defineproperty(obj, key, {
        get: function () {
          return val;
        },
        set: function (newval) {
          if (newval === val) {
            return; 
          } else {
            val = newval;
            console.log(val); // 方便看效果
          }
        }
      });
    }

    function observe (obj, vm) {
      object.keys(obj).foreach(function (key) {
        definereactive(vm, key, obj[key]);
      });
    }
  </script>

</body>
</html>

7. 订阅/发布模式(subscribe & publish)

text属性变化了,set方法触发了,但是文本节点的内容没有变化。如何才能让同样绑定到text的文本节点也同步变化呢?这里有一个知识点:订阅发布模式,订阅发布模式又称为观察者模式,定义一种一对多的关系,让多个观察者同时监听某一个主题对象,这个主题对象的状态发生改变时就会通知所有的观察者对象。
发布者发出通知 => 主题对象收到通知并推送给订阅者 => 订阅者执行相应的操作

// 一个发布者 publisher,功能就是负责发布消息 - publish
    var pub = {
      publish: function () {
        dep.notify();
      }
    }

    // 多个订阅者 subscribers, 在发布者发布消息之后执行函数
    var sub1 = { 
      update: function () {
        console.log(1);
      }
    }
    var sub2 = { 
      update: function () {
        console.log(2);
      }
    }
    var sub3 = { 
      update: function () {
        console.log(3);
      }
    }

    // 一个主题对象
    function dep() {
      this.subs = [sub1, sub2, sub3];
    }
    dep.prototype.notify = function () {
      this.subs.foreach(function (sub) {
        sub.update();
      });
    }

    // 发布者发布消息, 主题对象执行notify方法,进而触发订阅者执行update方法
    var dep = new dep();
    pub.publish();

不难看出,这里的思路还是很简单的: 发布者负责发布消息、 订阅者负责接收接收消息,而最重要的是主题对象,他需要记录所有的订阅这特消息的人,然后负责吧发布的消息通知给哪些订阅了消息的人。

所以,当set方法触发后做的第二件事情就是作为发布者发出通知: “我是属性text,我变了”。 文本节点作为订阅者,在接收到消息之后执行相应的更新动作。

8.双向绑定的实现

回顾一下,每当new一个vue,主要做了两件事情 ,第一监听数据:observe(data),第二是编译html, nodetofragment(id)
在监听数据的过程中,会为data中的每一个属性生成一个主题对象dep。
在编译html的过程中,会为每一个数据绑定相关的节点生成一个订阅者watcher,watcher会将自己添加到相应属性的dep中。
我们已经实现了:修改输入框内容 => 在事件回调函数中修改属性值 => 触 发属性的set方法。
接下来我们要实现的是:发出通知dep.notify() => 触发订阅者update方法 => 更新视图。
这里的关键逻辑是:如何将watch添加到关联属性的dep中。

function observe(obj, vm) {
 object.keys(obj).foreach(function(key) {
  definereactive(vm, key, obj[key])
 })
}

function definereactive(obj, key, val) {
 var dep = new dep()
 object.defineproperty(obj, key, {
  get: function() {
   if (dep.target) {
    // 添加订阅者watcher到主题对象dep
    dep.addsub(dep.target)
   }
   return val
  },
  set: function(newval) {
   if (newval === val) {
    return
   } else {
    val = newval
    // 作为发布者发出通知
    dep.notify()
    
   }
   
  }
 })
}

function dep () {
 this.subs = []
}

dep.prototype = {
 addsub: function(sub) {
  this.subs.push(sub)
 },
 notify: function() {
  this.subs.foreach(function(sub) {
   sub.update()
  })
 }
}

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.addeventlistener('input', function(e) {
     vm[name] = e.target.value
    })
    node.value = vm[name]
    node.removeattribute('v-model')
   }
  }
 }
 
 if (node.nodetype === 3) {
  if (reg.test(node.nodevalue)) {
   var name = regexp.$1
   name = name.trim()
   // node.nodevalue = vm[name]
   new watcher(vm, node, name)
  }
 }
}

function nodetofragment(node, vm) {
 var flag = document.createdocumentfragment()
 var child 
 while (child = node.firstchild) {
  compile(child, vm)
  flag.appendchild(child)
 }
 return flag
}

function watcher(vm, node, name){
 dep.target = this 
 this.vm = vm 
 this.node = node 
 this.name = name 
 this.update()
 dep.target = null
}

watcher.prototype = {
 update: function() {
  this.get()
  this.node.nodevalue = this.value
 },
 get: function() {
  this.value = this.vm[this.name]
 }
}

function vue(options) {
 this.data = options.data
 this.methods = options.methods
 var data = this.data 
 var el = options.el
 
 observe(data, this)
 
 var dom = nodetofragment(document.queryselector(el), this)
 
 document.queryselector(el).appendchild(dom)
}

var vm = new vue({
 el: '#app',
 data: {
  text: 123
 }
})

希望本文所述对大家vue.js程序设计有所帮助。

如对本文有疑问, 点击进行留言回复!!

相关文章:

验证码:
移动技术网