vue中的观察者模式和发布订阅者模式
观察者模式
目标者对象和观察者对象有相互依赖的关系,观察者对某个对象的状态进行观察,如果对象的状态发生改变,就会通知所有依赖这个对象的观察者,
目标者对象 Subject,拥有方法:添加 / 删除 / 通知 Observer;
观察者对象 Observer,拥有方法:接收 Subject 状态变更通知并处理;
目标对象 Subject 状态变更时,通知所有 Observer。
Vue中响应式数据变化是观察者模式 每个响应式属性都有dep,dep存放了依赖这个属性的watcher,watcher是观测数据变化的函数,如果数据发生变化,dep就会通知所有的观察者watcher去调用更新方法。因此, 观察者需要被目标对象收集,目的是通知依赖它的所有观察者。那为什么watcher也要存放dep呢?是因为当前正在执行的watcher需要知道此时是哪个dep通知了自己。
class Observer {
// 需要对value的属性描述重新定义
constructor(value) {
this.walk(value); // 初始化的时候就对数据进行监控
}
walk(data) {
Object.keys(data).forEach((key) => {
defineReactive(data, key, data[key]);
});
}
}
function defineReactive(data, key, value) {
// value 可能是一个对象,要递归劫持,所以数据不能嵌套太深
observe(value);
let dep = new Dep();
Object.defineProperty(data, key, {
get() {
// 如果有 watcher,就让 watcher 记住 dep,防止产生重复的 dep, 同时 dep 也收集此 watcher
if (Dep.target) {
dep.depend();
}
return value;
},
set(newVal) {
// 数据没变动则不处理
if (value === newVal) return;
observe(newVal); // 如果新值是个对象,递归拦截
value = newVal; // 设置新的值
dep.notify(); // 通知收集的 watcher 去更新
},
});
}
function observe(data) {
// 不是对象则不处理,isObject是用来判断是否为对象的函数
if (Object.prototype.toString.call(data)!== '[object Object]') return;
// 通过类来实现数据的观测,方便扩展,生成实例
return new Observer(data);
}
observe(data)
class Dep {
static target = null
constructor() {
this.id = id++;
this.subs = []; // 存放依赖的watcher
}
depend() {
// 让正在执行的watcher记录dep,同时dep也会记录watcher
Dep.target.addDep(this);
}
addSub(watcher) {
// 添加观察者对象
this.subs.push(watcher);
}
notify() {
// 触发观察者对象的更新方法
this.subs.forEach((watcher) => watcher.update());
}
}
class Watcher {
constructor(vm, exprOrFn) {
this.vm = vm;
this.deps = [];
// 用来去重,防止多次取同一数据时存入多个相同dep
this.depId = new Set();
// exprOrFn是updateComponent
this.getter = exprOrFn;
// 更新页面
this.get();
}
get() {
Dep.target = watcher; // 取值之前,收集 watcher
this.getter.call(this.vm); // 调用updateComponent更新页面
Dep.target = null; // 取值完成后,将 watcher 删除
}
// dep.depend执行时调用
addDep(dep) {
let id = dep.id;
let has = this.depId.has(id);
if (!has) {
this.depId.add(id);
// watcher存放dep
this.deps.push(dep);
// dep存放watcher
dep.addSub(this);
}
}
// 更新页面方法,dep.notify执行时调用
update() {
this.get(); // 一修改数据就渲染更新
}
}
function mountComponent(vm) {
// 渲染更新页面
let updateComponent = () => {
let vnode = vm._render(); // 生成虚拟节点 vnode
vm._update(vnode); // 将vnode转为真实节点
};
// 每个组件都要调用一个渲染 watcher
new Watcher(vm, updateComponent);
}
mountComponent(vm)
发布订阅模式
基于一个事件中心,接收通知的对象是订阅者,需要 先订阅某个事件,触发事件的对象是发布者,发布者通过触发事件,通知各个订阅者。 js中事件绑定,就是发布订阅模式
发布订阅模式相比观察者模式多了个事件中心,订阅者和发布者不是直接关联的。
vue中的事件总线就是使用的发布订阅模式
// 事件总线
class Bus {
constructor() {
// 用来记录事件和监听该事件的数组
this.listeners = {};
}
// 添加指定事件的监听者
$on(eventName, handler) {
this.listeners[eventName].add(handler);
}
// 取消监听事件
$off(eventName, handler) {
this.listeners[eventName].delete(handler);
}
// 触发事件
$emit(eventName, ...args) {
this.listeners[eventName].forEach((fn) => fn(...args));
}
}
事件总线的具体使用可以查看这篇vue之事件总线
观察者模式和发布订阅模式的区别
目标和观察者之间是互相依赖的。
发布订阅模式是由统一的调度中心调用,发布者和订阅者不知道对方的存在。
vue中的观察者模式和发布订阅者模式的更多相关文章
- [JS设计模式]:观察者模式(即发布-订阅者模式)(4)
简介 观察者模式又叫发布---订阅模式,它定义了对象间的一种一对多的关系,让多个观察者对象同时监听某一个主题对象,当一个对象发生改变时,所有依赖于它的对象都将得到通知. 举一个现实生活中的例子,例如小 ...
- js中的观察者模式与发布者/订阅者模式的区别?
- 浅谈vue响应式原理及发布订阅模式和观察者模式
一.Vue响应式原理 首先要了解几个概念: 数据响应式:数据模型仅仅是普通的Javascript对象,而我们修改数据时,视图会进行更新,避免了繁琐的DOM操作,提高开发效率. 双向绑定:数据改变,视图 ...
- 观察者模式 vs 发布-订阅模式
我曾经在面试中被问道,_“观察者模式和发布订阅模式的有什么区别?” _我迅速回忆起“Head First设计模式”那本书: 发布 + 订阅 = 观察者模式 “我知道了,我知道了,别想骗我” 我微笑着回 ...
- js之观察者模式和发布订阅模式区别
观察者模式(Observer) 观察者模式指的是一个对象(Subject)维持一系列依赖于它的对象(Observer),当有关状态发生变更时 Subject 对象则通知一系列 Observer 对象进 ...
- 观察者模式(Observer)和发布-订阅者模式(Publish/Subscribe)区别
观察者模式:定义一对多的关系,让多个观察对象同时监听某一个主题对象,主题对象状态发生变化就通知所有观察者对象.所以它是由两类对像组成:Subject主题+Observer观察者.主题发布事件,观察者通 ...
- “一切都是消息”--MSF(消息服务框架)之【发布-订阅】模式
在上一篇,“一切都是消息”--MSF(消息服务框架)之[请求-响应]模式 ,我们演示了MSF实现简单的请求-响应模式的示例,今天来看看如何实现[发布-订阅]模式.简单来说,该模式的工作过程是: 客户端 ...
- “一切都是消息”--iMSF(即时消息服务框架)之【发布-订阅】模式
MSF的名字是 Message Service Framework 的简称,由于目前框架主要功能在于处理即时(immediately)消息,所以iMSF就是 immediately Message S ...
- ActiveMQ发布-订阅消息模式
一.订阅杂志我们很多人都订过杂志,其过程很简单.只要告诉邮局我们所要订的杂志名.投递的地址,付了钱就OK.出版社定期会将出版的杂志交给邮局,邮局会根据订阅的列表,将杂志送达消费者手中.这样我们就可以看 ...
- (三)ActiveMQ之发布- 订阅消息模式实现
一.概念 发布者/订阅者模型支持向一个特定的消息主题发布消息.0或多个订阅者可能对接收来自特定消息主题的消息感兴趣.在这种模型下,发布者和订阅者彼此不知道对方.这种模式好比是匿名公告板.这种模式被概括 ...
随机推荐
- 时钟频率(HZ)与数据传输速率(bit/s)的关系-转载
(24条消息) 时钟频率(HZ)与数据传输速率(bit/s)的关系_子曰小玖的博客-CSDN博客_速率和频率的关系 时钟频率(HZ)与数据传输速率(bit/s)两者是相同的概念.举例:IIC传输位速率 ...
- navicat 远程连接不上mysql
1 查看是否开启远程连接(拿root用户举例) use mysql; select host, user from user; 以上便是开启远程连接,如果依旧不能连接,参考如下: grant all ...
- GPS时钟装置,北斗卫星授时,ntp校时服务器,网络时间服务器
GPS时钟装置,北斗卫星授时,ntp校时服务器,网络时间服务器 一.GPS时钟装置产品特点:技术交流-岳峰 15901092122:Q522508213 GPS时钟装置是针对计算机.自动化装置等进行校 ...
- STM32中断调试中遇到的问题
STM32应用过程中遇到的问题 实现功能: 1.自动流水灯:在LED1~LED4上实现自动流水灯,流水间隔时间为200ms/bit,然后通过按键KEY1改变流水灯的速度,每次按键间隔时间增加200ms ...
- 学习笔记||使用Vue时踩过的坑1.0
vue介绍:https://cn.vuejs.org/v2/guide/ 1.安装npm install时,长时间停留在fetchMetadata: sill mapToRegistry uri ht ...
- 使用LitJson输出格式化json文件到本地
百度上搜了半天,竟然没有C#使用LitJson格式化输出的例子,全都是Newtonsoft.Json的,最后在litjson的官网找到了方法. 给大家分享一下: https://litjson.net ...
- applicationContext.xml及springMVC.xml详解
在前面的web.xml详解里,我们引入applicationContext.xml和springMVC.xml两个配置文件, 前者是spring 全局配置文件,用来控制spring 特性的, 后者则是 ...
- 项目实训 DAY14
今天修改了一下PNN使之可以运行直接生成图片,而不是敲两段命令行.首先是使用python中subprocess启动新进程来达到命令行输命令的效果,即生成xx.pdf:再用os.unlink将中间品删除 ...
- (论文笔记)Learning Deep Structured Semantic Models for Web Search using Clickthrough Data
利用点击数据学习web搜索的深度学习模型 [总结] 该模型可以得到query和item的低维度向量表示,也可以得到二者的余弦语义相似度. 学习过程是通过最大化后验概率的极大似然估计得到的参数. ...
- go的相关包time、os、rand、fmt
time 1.time包 2.time.Time类型, 用来表示时间 3.取当前时间, now := time.Now() 4.time.Now().Day(),time.Now().Minute() ...