观察者模式(Observer Pattern)

观察者模式(Observer Pattern)

定义

  • 观察者模式定义了一种一对多的依赖关系,让多个观察者对象同时监听某一个主题对象。这个主题对象在状态发生变化时,会通知所有观察者对象,使它们能够自动更新自己。
  • 别名:模型-视图模式、源-收听者模式、从属者模式、发布-订阅模式。
  • 观察者模式通过解耦观察者和被观察者,使得它们可以独立地变化和复用。
  • 是行为型设计模式。

适用场景

  1. 当一个抽象模型包含两个方面内容,其中一个方面依赖于另一个方面;
  2. 其他一个或多个对象的变化依赖于另一个对象的变化;
  3. 实现类似于广播机制的功能,无需知道具体收听者,只需分发广播,系统中感兴趣的对象会自动接收该广播;
  4. 多层级嵌套使用,形成一种链式触发机制,使得事件具备跨域(跨越两种观察者类型)通知。

标准示例

在这里插入图片描述

观察者模式通常包含以下几个角色:
Subject(主题/被观察者):维护一个观察者列表,并提供注册、移除和通知观察者的方法。
Observer(观察者):定义一个更新接口,以便在得到主题通知时更新自己。
ConcreteSubject(具体主题):具体的被观察者,在内部状态改变时,通知所有注册的观察者。
ConcreteObserver(具体观察者):实现抽象观察者的定义,以便在得到主题状态更改通知时更新自身。

示例代码:
ISubject 抽象主题

/*** 抽象主题*/
public interface ISubject<E> {boolean attach(IObserver<E> observer);boolean detach(IObserver<E> observer);void notify(E event);
}

IObserver 抽象观察者

/*** 抽象观察者*/
public interface IObserver<E> {/*** 更新* @param event*/void update(E event);
}

ConcreteSubject 具体主题

/*** 具体主题*/
public class ConcreteSubject<E> implements ISubject<E>{private List<IObserver<E>> observers = new ArrayList<IObserver<E>>();public boolean attach(IObserver<E> observer) {return !this.observers.contains(observer) && this.observers.add(observer);}public boolean detach(IObserver<E> observer) {return this.observers.remove(observer);}public void notify(E event) {for(IObserver<E> observer:observers){observer.update(event);}}
}

ConcreteObserver 具体观察者

public class ConcreteObserver<E> implements IObserver<E>{public void update(E event) {System.out.println(this.getClass().getSimpleName() + " receive event : " + event);}
}

客户端调用代码:

public class ClientTest {public static void main(String[] args) {//被观察者ISubject<String> subject = new ConcreteSubject<String>();//观察者IObserver<String> observer = new ConcreteObserver<String>();//被观察者注册subject.attach(observer);//通知subject.notify("hello");}
}

输出执行结果:

ConcreteObserver receive event : hello

接下来,我们使用jdk中提供的观察者模式接口来实现一个如下场景:
明星发声,粉丝关注并收听。

Idol 明星主题抽象类

/*** 抽象主题者——偶像*/
public interface Idol {void posting(String message);
}

ConcreteIdol 明星主题具体类

/*** 具体偶像类*/
@Data
public class ConcreateIdol extends Observable implements Idol{private String name;public ConcreateIdol(String name){this.name = name;}public void posting(String message) {System.out.println("【明星发布】 "+message);setChanged();notifyObservers(message);}
}

Fans粉丝具体类

/*** 观察者*/
public class Fans implements Observer {private String idol;private String selfName;public Fans(String idol,String selfName){this.idol = idol;this.selfName = selfName;}public void update(Observable o, Object arg) {System.out.println("【粉丝"+this.selfName+"收到通知】 " +  "收到偶像消息:"+ arg);}
}

JDK中,java.util.Observable 类,相当于IObserver的存在,所以,我们的Fans类,只要继承Observable即可。在发布的被监听方法中,只要调用如下两个方法,可以实现发布。

setChanged();
notifyObservers();

而监听者实现类 Fans,只需要实现jdk中的 java.util.Observer 接口。其中的 update 方法,即为订阅到发布消息后的回调方法。
ClientTest 客户端调用类:

public class ClientTest {public static void main(String[] args) {ConcreateIdol zhazhahui = new ConcreateIdol("渣渣辉");zhazhahui.addObserver(new Fans(zhazhahui.getName(),"小明明"));zhazhahui.addObserver(new Fans(zhazhahui.getName(),"小甜甜"));String message = "大家好,我是"+zhazhahui.getName()+",欢迎体验我代言的新版砍怪看到手抽筋大型即使PK古装仙侠网游!";zhazhahui.posting(message);}
}

执行后的结果输出为:

【明星发布】 大家好,我是渣渣辉,欢迎体验我代言的新版砍怪看到手抽筋大型即使PK古装仙侠网游!
【粉丝小甜甜收到通知】 收到偶像消息:大家好,我是渣渣辉,欢迎体验我代言的新版砍怪看到手抽筋大型即使PK古装仙侠网游!
【粉丝小明明收到通知】 收到偶像消息:大家好,我是渣渣辉,欢迎体验我代言的新版砍怪看到手抽筋大型即使PK古装仙侠网游!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/42068.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

鼠标宏怎么设置?6款鼠标自动点击器强推,游戏玩家专用!(2024全)

随着电子游戏和日常应用的不断发展&#xff0c;我们经常会遇到一些重复性的任务或操作。而在这种情况下&#xff0c;鼠标宏以其自动化的特点成为了许多玩家和使用者的利器之一。如果你正在寻找如何设置鼠标宏来简化操作并提高效率&#xff0c;那么你来对地方了。在本文中&#…

【Java]认识泛型

包装类 在Java中&#xff0c;由于基本类型不是继承自Object&#xff0c;为了在泛型代码中可以支持基本类型&#xff0c;Java给每个基本类型都对应了一个包装类型。 除了 Integer 和 Character&#xff0c; 其余基本类型的包装类都是首字母大写。 泛型 泛型是在JDK1.5引入的…

ASAN排查程序中内存问题使用总结

简介 谷歌有一系列Sanitizer工具&#xff0c;可用于排查程序中内存相关的问题。常用的Sanitizer工具包括&#xff1a; Address Sanitizer&#xff08;ASan&#xff09;&#xff1a;用于检测内存使用错误。Leak Sanitizer&#xff08;LSan&#xff09;&#xff1a;用于检测内存…

【9-2:RPC设计】

RPC 1. 基础1.1 定义&特点1.2 具体实现框架1.3 应用场景2. RPC的关键技术点&一次调用rpc流程2.1 RPC流程流程两个网络模块如何连接的呢?其它特性RPC优势2.2 序列化技术序列化方式PRC如何选择序列化框架考虑因素2.3 应用层的通信协议-http什么是IO操作系统的IO模型有哪…

Vortex GPGPU的硬件设计和代码结构分析

文章目录 前言一、GPGPU是什么&#xff1f;1.1 GPU和GPGPU之间的差异1.2 GPU和CPU之间的集成方式1.3 GPU包含什么&#xff08;列举和VMIPS向量体系结构的差异&#xff09; 二、Vortex GPGPU是什么&#xff1f;2.1 Vortex GPGPU的技术边界和验证环境2.2 Vortex GPGPU的指令集设计…

安卓稳定性之crash详解

目录 前言一、Crash 的基本原理二、Crash 分析思路三、实例分析四、预防措施五、参考链接 前言 在开发和测试 Android 应用程序时&#xff0c;遇到应用程序崩溃是很常见的情况。 Android 崩溃指的是应用程序因为异常或错误而无法正常执行&#xff0c;并且导致应用强制关闭。 一…

p11函数和递归

递归与迭代 求n的阶乘。&#xff08;不考虑溢出&#xff09; int Fac1(int n) {int i0;int ret1;for(i1;i<n;i){ret*i;}return ret; } int main(){//求n的阶乘int n0;int ret0;scanf("%d",&n);retFac1(n);printf("%d\n",ret);return 0; } int Fac…

unity知识点 专项四 一文彻底说清楚(锚点(anchor)、中心点(pivot)、位置(position)之间的关系)

一 概述 想要使UI控件在屏幕中达到正确的显示效果&#xff0c;比如自适应屏幕尺寸、固定边距等等&#xff0c;首先要理清楚几个基本概念和设置&#xff1a;锚点(anchor)、中心点(pivot)、位置(position)、UI缩放模式、父物件的transform设置 二 Anchor、Pivot与Position 2…

网络连接线相关问题

问题1&#xff1b; 直通线为什么两头都是T568B&#xff1f;是否可以两台T5568A&#xff1f;或者任意线序&#xff0c;只需两头一致&#xff1f; 不行&#xff0c;施工规范规定。&#xff08;原因&#xff1b;网线最长距离100m&#xff0c;实际用起来要把网线包管&#xff0c;走…

【分布式系统】Filebeat+Kafka+ELK 的服务部署

目录 一.实验准备 二.配置部署 Filebeat 三.配置Logstash 四.验证 一.实验准备 结合之前的博客中的实验 主机名ip地址主要软件es01192.168.80.101ElasticSearches02192.168.80.102ElasticSearches03192.168.80.103ElasticSearch、Kibananginx01192.168.80.104nginx、Logs…

iperf3: error - unable to connect to server: No route to host

1.确认iperf3版本是否统一。 2.确认防火墙是否关闭。 关闭防火墙 : systemctl stop firewalld 查看防火墙状态: systemctl status firewalld 3.重新建起链接

自动驾驶算法———车道检测(一)

“ 在本章中&#xff0c;我将指导您构建一个简单但有效的车道检测管道&#xff0c;并将其应用于Carla 模拟器中捕获的图像。管道将图像作为输入&#xff0c;并产生车道边界的数学模型作为输出。图像由行车记录仪&#xff08;固定在车辆挡风玻璃后面的摄像头&#xff09;捕获。…

【ZIP压缩大揭秘】轻松掌握ZIP分卷压缩包的高效解压秘籍!

在这个信息爆炸的时代&#xff0c;文件大小常常成为我们分享与存储的绊脚石。幸运的是&#xff0c;ZIP分卷压缩技术如同一把钥匙&#xff0c;巧妙地将庞然大物分解成小巧易管理的部分。但面对这一串分卷压缩包&#xff0c;你是否也曾迷茫于如何高效解压&#xff0c;恢复文件的完…

解码Python字符串:‘r‘、‘b‘、‘u‘和‘f‘前缀的全面指南

&#x1f4d6; 正文 1 字符串前加’r’ 表示原始字符串&#xff0c;消除转义 print(abc\nde) # abc # deprint(rabc\nde) # abc\nde在下面这个列子中&#xff0c;如果不在路径字符串前面加r那么&#xff0c;路径中的空格就会出现问题 print(rD:\01 programming\09python\py…

全志A527 T527 cat /proc/cupinfo没有Serial问题

1.前言 我们有些客户是使用cpuinfo节点去获取系统的cpuid的,如下: cat /proc/cupinfo processor : 0 BogoMIPS : 48.00 Features : fp asimd evtstrm aes pmull sha1 sha2 crc32 atomics fphp asimdhp cpuid asimdrdm lrcpc dcpop asimddp CPU impleme…

系统吃swap问题排查

目录 背景 问题 分析并解决 1.控制线程数 2.更换IO组件 3.Linux进程信息文件分析 总结加餐 参考文档 背景 隔壁业务组系统是简单的主从结构&#xff0c;写索引的服务(主)叫primary&#xff0c; 读索引并提供搜索功能的服务(从)叫replica。业务线同步数据并不是平滑的&…

static的理论学习

在说到static之前&#xff0c;需要先明确变量类型&#xff1a; 而在聊到变量类型之前我们可以将变量的两个属性好好学一学 变量的两个属性 作用域&#xff08;scope&#xff09;&#xff1a; 从内存的角度来看&#xff0c;就是变量存放在栈&#xff08;stack&#xff09;中&…

TypeError: Cannot read properties of null (reading ‘nextSibling‘)

做项目用的Vue3Vite, 在画静态页面时&#xff0c;点击菜单跳转之后总是出现如下报错&#xff0c;百思不得其解。看了网上很多回答&#xff0c;也没有解决问题&#xff0c;然后试了很多方法&#xff0c;最后竟然发现是template里边没有结构的原因。。。 原来我的index.vue是这样…

ELK+Filebeat+Kafka+Zookeeper

本实验基于ELFK已经搭好的情况下 ELK日志分析 架构解析 第一层、数据采集层 数据采集层位于最左边的业务服务器集群上&#xff0c;在每个业务服务器上面安装了filebeat做日志收集&#xff0c;然后把采集到的原始日志发送到Kafkazookeeper集群上。第二层、消息队列层 原始日志发…

Matlab手搓线性回归-非正规方程法

原理&#xff1a;wxb&#xff0c;x是输入&#xff0c;求得的结果与真实值y求均方误差。 采用链式法则求导 参数更新&#xff0c;梯度下降法&#xff08;批量梯度下降&#xff09; 随机生成数据&#xff1a; m100&#xff1b;生成100个数据&#xff0c;并添加随机噪声 clear; …