一、总结

vue中有2种数据绑定的方式:

  1. 单向数据绑定(v-bind):数据只能从data流向页面;
  2. 双向数据绑定(v-model):数据不仅能从data流向页面,还可以从页面流向data;

备注:

  • 双向数据绑定一般都应用在表单类(输入类)元素上 (如:input、select等);
  • v-model:value可以简写为v-model,因为v-model默认收集的就是value的值;

二、分析

单向数据绑定:就是把Model绑定到View,当我们用JavaScript代码更新Model时,View就会自动更新双向绑定就很容易联想到了;

双向数据绑定:在单向绑定的基础上,用户更新了ViewModel的数据也自动被更新了

三、 双向绑定的原理:

Vue 是数据双向绑定的框架,双向绑定由三个重要部分构成:

  • 数据层(Model):应用的数据及业务逻辑
  • 视图层(View):应用的展示效果,各类UI组件
  • 业务逻辑层(ViewModel):框架封装的核心,它负责将数据与视图关联起来

理解ViewModel,它的主要职责就是:

  • 数据变化后更新视图
  • 视图变化后更新数据

它还有两个主要部分组成

  • 监听器(Observer):对所有数据的属性进行监听
  • 解析器(Compiler):对每个元素节点的指令进行扫描跟解析,根据指令模板替换数据,以及绑定相应的更新函数

四、实现双向绑定

  1. new vue()首先执行初始化,对data执行响应式处理,这个过程发生在observe中;
  2. 同时对模板执行编译,找到其中动态绑定的数据,从data中获取并初始化视图,这个过程发生在Compile中;
  3. 同时定义一个更新函数和Watcher,将来对应数据变化时Watcher会调用更新函数;
  4. 由于data中的某个key在一个视图中可能出现多次,所以每个key都需要一个管家Dep来管理多个Watcher
  5. 将来data中数据一旦发生变化,会首先找到对应的Dep,通知所有Watcher执行更新函数;

五、实现

先来一个构造函数:执行初始化,对data执行响应化处理

class Vue {  constructor(options) {  this.$options = options;  this.$data = options.data;  // 对data选项做响应式处理  observe(this.$data);  // 代理data到vm上  proxy(this);  // 执行编译  new Compile(options.el, this);  }
}  

data选项执行响应化具体操作

function observe(obj) {  if (typeof obj !== "object" || obj == null) {  return;  }  new Observer(obj);
}  class Observer {  constructor(value) {  this.value = value;  this.walk(value);  }  walk(obj) {  Object.keys(obj).forEach((key) => {  defineReactive(obj, key, obj[key]);  });  }
}  

 编译Compile

对每个元素节点的指令进行扫描跟解析,根据指令模板替换数据,以及绑定相应的更新函数

class Compile {  constructor(el, vm) {  this.$vm = vm;  this.$el = document.querySelector(el);  // 获取dom  if (this.$el) {  this.compile(this.$el);  }  }  compile(el) {  const childNodes = el.childNodes;   Array.from(childNodes).forEach((node) => { // 遍历子元素  if (this.isElement(node)) {   // 判断是否为节点  console.log("编译元素" + node.nodeName);  } else if (this.isInterpolation(node)) {  console.log("编译插值⽂本" + node.textContent);  // 判断是否为插值文本 {{}}  }  if (node.childNodes && node.childNodes.length > 0) {  // 判断是否有子元素  this.compile(node);  // 对子元素进行递归遍历  }  });  }  isElement(node) {  return node.nodeType == 1;  }  isInterpolation(node) {  return node.nodeType == 3 && /\{\{(.*)\}\}/.test(node.textContent);  }
}  

 收集依赖

视图中会用到data中某key,这称为依赖。同一个key可能出现多次,每次都需要收集出来用一个Watcher来维护它们,此过程称为依赖收集多个Watcher需要一个Dep来管理,需要更新时由Dep统一通知;

实现思路

  1. defineReactive是为每一个key创建一个Dep实例;
  2. 初始化视图时读取某个key,例如name1,创建一个Wathcher1;
  3. 由于触发name1的getter方法,便将Wathcher1添加到name1的对应的Dep中;
  4. 当name1更新,setter触发时,便可通过对应Dep通知其管理所有Watncer更新;
// 负责更新视图
class Watcher {  constructor(vm, key, updater) {  this.vm = vm  this.key = key  this.updaterFn = updater  // 创建实例时,把当前实例指定到Dep.target静态属性上  Dep.target = this  // 读一下key,触发get  vm[key]  // 置空  Dep.target = null  }  // 未来执行dom更新函数,由dep调用的  update() {  this.updaterFn.call(this.vm, this.vm[this.key])  }
}  

声明Dep

class Dep {  constructor() {  this.deps = [];  // 依赖管理  }  addDep(dep) {  this.deps.push(dep);  }  notify() {   this.deps.forEach((dep) => dep.update());  }
}  

创建watcher时触发getter

class Watcher {  constructor(vm, key, updateFn) {  Dep.target = this;  this.vm[this.key];  Dep.target = null;  }
}  

依赖收集,创建Dep实例

function defineReactive(obj, key, val) {  this.observe(val);  const dep = new Dep();  Object.defineProperty(obj, key, {  get() {  Dep.target && dep.addDep(Dep.target);// Dep.target也就是Watcher实例  return val;  },  set(newVal) {  if (newVal === val) return;  dep.notify(); // 通知dep执行更新方法  },  });
}

结束~~

vue单向数据绑定和双向数据绑定相关推荐

  1. Vue 响应式原理(双向数据绑定) 怎样实现 响应式原理?

    Vue 响应式原理(双向数据绑定) 怎样实现 响应式原理? 我们在Vue里面,定义在Data里的属性,叫做响应式属性. 每一个vue组件被创建的时候,同时还有一个对象被创建出来了,这个对象我们是看不到 ...

  2. vue面试核心,双向数据绑定,数据代理,数据劫持,发布订阅,数据编译,看这个demo就够了

    vue面试核心,双向数据绑定,数据代理,数据劫持,发布订阅,数据编译,看这个demo就够了. 不在怕面试官,你给我怼过去.复制到html文档,浏览器运行即可. mvvm原理剖析 {{singer}}

  3. Vue.js 基础语法 入门语句 Vue学习笔记 v-model 双向数据绑定

    Vue.js 基础语法,入门语句,Vue学习笔记 学习网站:https://www.bilibili.com/video/BV15741177Eh vue 的体验 响应式:数据一旦改变,视图就会响应改 ...

  4. Vue绑定事件,双向数据绑定,只是循环没那么简单

    v-on对象处理 <p @mouseover = "doTish" @mouseout = "doThat"> 对象形式 </p>< ...

  5. vue的v-model的双向数据绑定原理

    vue的双向数据绑定是一个很有意思的特性:知道其原理方能运用自如 <div id="demo"><h1>双向绑定机制</h1><!--表单 ...

  6. Vue单向绑定与双向绑定解析

    在讲绑定之前我们要理解MVVM框架,这对我们理解单向以及双向绑定有很大的帮助. 一 MVVM框架 M : model(模型) 数据保存 V : view(视图) 用户界面 VM : ViewModel ...

  7. vue单向数据流与双向绑定两者区别

    1.Vue 在不同组件间强制使用单向数据流,父组件可以向子组件传递数据,但是子组件不能直接修改父组件的状态. 2.数据的双向绑定 主要由MVVM框架实现,主要由三部分组成View.ViewModel和 ...

  8. Vue数据绑定(单向绑定,双向绑定)

    Vue中有两种数据绑定(单向数据绑定和双向数据绑定) 1.单向数据绑定(v-bind):数据只能从data流向页面. 单像数据绑定:<input type="text" na ...

  9. Vue数据响应式与双向数据绑定原理区分

    很多小伙伴搞不清楚数据响应式与双向数据绑定这两个原理,还有人自然的认为它们两者是相同的,接下来我们来看看它们二者的区分吧! 数据响应式是Vue的核心特性之一,而双向数据绑定是通过指令或修饰符实现的一种 ...

  10. Vue中从v-model,model,.sync到双向数据传递,再到双向数据绑定

    前面的话 虽然v-model和.sync之间的区别,是非常明显的,但是通过这个去研究下去,就有点意思了.首先说明v-model是.sync的一种特殊情况,也就是当prop是value,vm.$emit ...

最新文章

  1. python如何调用文件进行换位加密_python 换位密码算法的实例详解
  2. C/C++关键字 static 和 const
  3. Spring Boot——基于AOP的HTTP操作日志解决方案
  4. 乔布斯和任正非相比,谁更厉害?
  5. SVN:请求不到主机,应该如何解决?
  6. 搜索引擎基础概念(1)—— 倒排索引
  7. redmine测试使用小结
  8. 小样本不符合正态_金莎相亲对象不符合择偶标准,男生单眼皮小眼睛,但她还是沦陷了...
  9. 算法:回溯十七 Combination Sum III挑选数组中规定个数元素的和为指定数
  10. 用linux运行vasp,科学网—VASP使用 - 李继存的博文
  11. DOM4J及SAXReader解析xml文件数据
  12. Java轻量级缓存Ehcache与SpringBoot整合
  13. codeforces 697C Lorenzo Von Matterhorn(二叉树LCA)
  14. win10定时开机(Win10定时开机关机)
  15. DirectX 9.0c游戏开发手记之RPG编程自学日志之11: 题外话
  16. 什么是 JavaBeans ?
  17. Linux达人养成计划 一
  18. 浅析FeiQ解密函数的使用
  19. 图像处理3-经典空间域增强——空域滤波
  20. 【多式联运】基于matlab粒子群结合遗传算法求解陆海空多式联运问题【含Matlab源码 1963期】

热门文章

  1. Excel的DATEDIF函数及其用法实例——求日期之间的间隔
  2. Mysql 创建数据库\添加用户\用户授权
  3. Python是个什么鬼?为什么那么多工作“会Python优先”?
  4. 怎么关闭计算机安全,电脑中安全防护怎么关闭
  5. QT自定义按钮类(自定义图片,形状,点击特效)
  6. 线性代数(1):行列式和展开式
  7. snapchat忘记账户_如何删除您的Snapchat帐户
  8. Windows7 Ghost 旗舰装机版
  9. 使用modelsim仿真时出现“vopt returned success but vsim could not find a design to simulate!”错误
  10. 别细看|请收藏|堆垛机故障大全及解决办法