Java观察者模式实现方式与测试方法

一、实现方式

  1. 自定义实现
    通过手动定义SubjectObserver接口,实现一对多依赖关系:
// 观察者接口
public interface Observer {void update(float temp, float humidity, float pressure);
}
// 主题接口
public interface Subject {void registerObserver(Observer o);void removeObserver(Observer o);void notifyObservers();
}
// 具体主题类(天气数据)
public class WeatherData implements Subject {private List observers = new ArrayList<>();private float temperature, humidity, pressure;@Overridepublic void registerObserver(Observer o) {observers.add(o);}@Overridepublic void removeObserver(Observer o) {observers.remove(o);}@Overridepublic void notifyObservers() {for (Observer o : observers) {o.update(temperature, humidity, pressure);}}public void setMeasurements(float temp, float hum, float press) {temperature = temp;humidity = hum;pressure = press;notifyObservers();}
}

特点:灵活可控,但需自行处理线程安全和内存管理。


  1. 使用JDK内置类(ObservableObserver
    JDK提供的Observable类和Observer接口简化实现,但需注意其设计缺陷(如需手动调用setChanged()):
// 具体主题类
public class WeatherData extends Observable {private float temperature, humidity, pressure;public void setMeasurements(float temp, float hum, float press) {temperature = temp;humidity = hum;pressure = press;setChanged();  // 标记状态变化notifyObservers(new MeasurementData(temp, hum, press)); // 推模型}
}
// 具体观察者类
public class CurrentConditionsDisplay implements Observer {@Overridepublic void update(Observable o, Object arg) {if (arg instanceof MeasurementData) {MeasurementData data = (MeasurementData) arg;System.out.println("温度:" + data.getTemp());}}
}

注意事项:

  • notifyObservers()需在setChanged()后调用,否则不触发通知。
  • 观察者需通过参数arg获取数据(推模型)或从主题拉取数据(拉模型)。

  1. 高级实现(异步与线程安全)
    通过线程池异步通知观察者,避免阻塞主题线程:
public class AsyncSubject extends Subject {private ExecutorService executor = Executors.newFixedThreadPool(4);@Overridepublic void notifyObservers() {for (Observer o : observers) {executor.submit(() -> o.update(...));}}
}

优势:提升并发性能,避免单线程阻塞。


二、测试方法

  1. 单元测试(Mockito框架)
    使用Mockito模拟观察者行为,验证通知逻辑:
@Test
public void testObserverNotification() {// 1. 创建Mock观察者Observer mockObserver = Mockito.mock(Observer.class);// 2. 注册观察者到主题WeatherData weatherData = new WeatherData();weatherData.registerObserver(mockObserver);// 3. 触发状态变化weatherData.setMeasurements(25.5f, 65, 1013.1f);// 4. 验证观察者方法是否被调用Mockito.verify(mockObserver).update(25.5f, 65, 1013.1f);
}

关键点:

  • 使用when()设置Mock对象行为(如异常抛出)。
  • 使用verify()验证方法调用次数和参数。

  1. 集成测试(多线程场景)
    测试观察者模式在并发环境下的稳定性:
@Test
public void testConcurrentObservers() {WeatherData weatherData = new WeatherData();Observer observer1 = new CurrentConditionsDisplay();Observer observer2 = new StatisticsDisplay();// 多线程注册观察者Thread t1 = new Thread(() -> weatherData.registerObserver(observer1));Thread t2 = new Thread(() -> weatherData.registerObserver(observer2));t1.start();t2.start();// 等待线程结束t1.join();t2.join();// 触发通知并验证weatherData.setMeasurements(30f, 70, 1012.5f);// 验证两个观察者均被通知
}

注意事项:

  • 使用synchronized或并发集合(如CopyOnWriteArrayList)保证线程安全。
  • 测试观察者是否因循环依赖导致死锁。

  1. 性能测试
    评估观察者数量对通知效率的影响:
@Test
public void testObserverPerformance() {WeatherData weatherData = new WeatherData();int observerCount = 1000;for (int i = 0; i < observerCount; i++) {weatherData.registerObserver(new SimpleObserver());}long startTime = System.currentTimeMillis();weatherData.setMeasurements(25f, 60, 1013f);long endTime = System.currentTimeMillis();// 验证耗时是否在合理范围内Assert.assertTrue(endTime - startTime < 1000); // 1秒内完成
}

优化方向:

  • 使用异步通知减少阻塞。
  • 限制观察者数量或引入优先级队列。

三、常见问题与解决方案

  1. 内存泄漏:
    • 问题:未从主题移除观察者,导致无法被GC回收。
    • 解决方案:在观察者销毁时调用removeObserver(),或使用弱引用存储观察者。
  2. 循环依赖:
    • 问题:主题与观察者互相依赖,导致栈溢出。
    • 解决方案:通过中介者模式解耦。
  3. 线程安全:
    • 问题:多线程环境下注册/移除观察者时数据不一致。
    • 解决方案:使用synchronized或并发集合(如CopyOnWriteArrayList)。

四、总结

  • 实现选择:优先自定义实现以避免JDK类的缺陷,复杂场景可结合异步通知。
  • 测试重点:验证通知逻辑、线程安全及性能。
  • 工具推荐:Mockito用于单元测试,线程池和并发集合用于多线程场景。

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

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

相关文章

leetGPU解题笔记(1)

1.题面 题目要求 向量加法 实现一个程序&#xff0c;在GPU上对两个包含32位浮点数的向量执行逐元素加法。该程序应接受两个长度相等的输入向量&#xff0c;并生成一个包含它们和的输出向量。 实现要求 禁止使用外部库 solve函数签名必须保持不变 最终结果必须存储在向量C中 示例…

5. JVM 的方法区

1. JVM介绍和运行流程-CSDN博客 2. 什么是程序计数器-CSDN博客 3. java 堆和 JVM 内存结构-CSDN博客 4. 虚拟机栈-CSDN博客 5. JVM 的方法区-CSDN博客 6. JVM直接内存-CSDN博客 7. JVM类加载器与双亲委派模型-CSDN博客 8. JVM类装载的执行过程-CSDN博客 9. JVM垃圾回收…

网络安全的基本练习

一.docker搭建 1.安装dockerapt-get install docker.io docker-compose2.编写配置文件&#xff08;注意路径正确&#xff09;vim /etc/systemd/system/docker.service.d/http-proxy.conf[Service] Environment"HTTP_PROXYhttp://科学上网访问的ip:端口" Environment&…

380. O(1) 时间插入、删除和获取随机元素

实现RandomizedSet 类&#xff1a; RandomizedSet() 初始化 RandomizedSet 对象 bool insert(int val) 当元素 val 不存在时&#xff0c;向集合中插入该项&#xff0c;并返回 true &#xff1b;否则&#xff0c;返回 false 。 bool remove(int val) 当元素 val 存在时&#xff…

【LeetCode Hot100 | 每日刷题】字母异位词分组

题目链接&#xff1a;49. 字母异位词分组 - 力扣&#xff08;LeetCode&#xff09; 题目&#xff1a; 给你一个字符串数组&#xff0c;请你将 字母异位词 组合在一起。可以按任意顺序返回结果列表。 示例 1: 输入: strs ["eat", "tea", "tan"…

docker 安装windows

目录 下载地址&#xff1a; 使用教程&#xff1a; docker compose 查看版本 测试启动 hello-world 报错1 The system cannot find the file specified&#xff1a; 检查 Docker Desktop 是否运行中 报错2HF_ENDPOINT 1. 临时解决方案&#xff08;当前终端会话有效&…

docker compose 和build

目录 docker compose 和build 的区别是什么&#xff1f; 核心差别&#xff1a; 1. docker build --platform linux/amd64 -f Dockerfile -t infiniflow/ragflow:nightly_lbg . 2. docker compose -f docker-compose-gpu.yml up -d 二者如何配合&#xff1f; 总结 docker …

裂变时刻:全球关税重构下的券商交易系统跃迁路线图(2025-2027)

——基于RWA清算、量子加密与实时非线性风控的下一代跨境基础设施核心事件锚定&#xff1a;特朗普于7月7日对14国启动分级关税制裁&#xff08;日韩25%、东南亚30%-40%、金砖关联国10%附加税&#xff09;&#xff0c;引发日元兑美元暴跌至144.47、铜价单日跳涨3.2%、散户单日交…

python爬虫初入门——基本库和写入方法

1.准备环境 python环境&#xff1a;3.10 2.常用库 1.请求库&#xff1a;实现 HTTP 请求操作 requests&#xff1a;基于 urllib 编写的&#xff0c;阻塞式 HTTP 请求库&#xff0c;发出一个请求&#xff0c;一直等待服务器响应后&#xff0c;程序才能进行下一步处理。seleni…

Sonar扫描C#代码配置

需要的工具 MSBuild、sonar-scanner-4.6.1.2450-windows、jdk1.8.0_181 下载地址&#xff1a;https://download.csdn.net/download/code12313/91315686 配置sonar的地址 一、环境变量配置 1.新建变量&#xff0c;nameSONAR_RUNNER_MSBUILD_HOME。valueD:\work\dev\dev_serve…

python 在运行时没有加载修改后的版本

陈旧的Python字节码 (.pyc 文件)&#xff1a;最常见的原因&#xff01;Python 会把你修改的 .py 文件编译成 .pyc 字节码来加速后续运行。有时&#xff0c;即使你修改了 .py 文件&#xff0c;系统可能仍然固执地加载旧的、未被删除的 .pyc 文件。1. 用“硬编码探针”强制验证# …

【会员专享数据】2013-2024年我国省市县三级逐年SO₂数值数据(Shp/Excel格式)

之前我们分享过2013-2024年全国范围逐年SO₂栅格数据&#xff08;可查看之前的文章获悉详情&#xff09;&#xff01;该数据来源于韦晶博士、李占清教授团队发布在国家青藏高原科学数据中心网站上的中国高分辨率高质量近地表空气污染物数据集。很多小伙伴拿到数据后反馈栅格数据…

出现SSL连接错误的原因和解决方案

介绍 SSL连接错误是一种常见但关键的问题&#xff0c;这可能会阻止客户端和服务器之间的安全连接。这些错误发生在TLS握手过程失败时&#xff0c;这意味着客户端和服务器无法建立安全的HTTPS连接。这种失败可以在SSL/TLS协商过程中的任何阶段发生&#xff0c;从初始协议协议到…

vue3 el-date-picker 保存后 日期减一问题

在使用 el-date-picker&#xff08;Element UI 的日期选择器组件&#xff09;时&#xff0c;如果你发现日期在保存到后台后自动减一&#xff0c;这通常是由于时区差异或者是时间格式解析问题导致的。这里有一些可能的解决方案&#xff1a;1. 检查前端发送的日期格式确保你在前端…

什么是IP关联?跨境卖家如何有效避免IP关联?

一位深圳卖家曾管理30个亚马逊店铺账号&#xff0c;某日清晨发现所有账号被批量封禁——原因竟是平台检测到这些账号长期共享同一IP地址&#xff0c;判定为“IP关联”。而在跨境领域如亚马逊、eBay、Shopee、TikTok等平台&#xff09;&#xff0c;对于IP关联的判定都是比较严格…

Redis集群方案——哨兵机制

Redis Sentinel&#xff08;哨兵&#xff09;是Redis官方提供的高可用性(HA)解决方案&#xff0c;用于管理Redis主从架构并实现自动故障转移。一、集群结构和作用哨兵是一个分布式系统&#xff0c;由多个哨兵节点组成&#xff1a;哨兵的作用如下&#xff1a;监控&#xff1a;Se…

1.2.3_2 TCP/IP模型

在这个视频中&#xff0c;我们要探讨TCP/IP模型。对于TCP/IP模型考试的要求是这样的&#xff0c;首先我们需要记住它各个层次的名称和顺序&#xff0c;以及我们需要了解TCP/IP 模型和OSI参考模型&#xff0c;它们在设计理念上有哪些区别&#xff0c;设计理念的区别又导致了TCP/…

EVOLVEpro安装使用教程-蛋白质语言模型驱动的快速定向进化

写在前面&#xff1a;蛋白质是生命活动的基石&#xff0c;其功能和序列之间的复杂关系长期以来吸引着科学家们的关注。尽管深度突变扫描等实验方法可以解析蛋白质突变的功能影响&#xff0c;但这些技术的应用范围局限于序列空间的一小部分。近年来&#xff0c;基于蛋白质语言模…

【Linux】Rocky Linux 清华镜像源安装 GitLab

使用清华镜像源安装 GitLab 地址&#xff1a;清华镜像源 1. 搜索 gitlab&#xff0c;我们选择 gitlab-ce 社区版进行安装 2. 新建 /etc/yum.repos.d/gitlab-ce.repo&#xff0c;内容为 注意&#xff1a;el$releasever 是清华镜像源内的文件夹版本 [gitlab-ce] nameGitlab C…

【龙泽科技】新能源汽车维护与动力蓄电池检测仿真教学软件【吉利几何G6】

产品简介新能源汽车维护与动力蓄电池检测仿真教学软件是依托《全国职业院校技能大赛》“新能源汽车维修”赛项中“新能源汽车维护与动力蓄电池检测” 竞赛模块&#xff0c;自主开发的一款仿真教学软件。软件采用仿真技术对车辆进行指定维护作业&#xff0c;并对动力蓄电池总成进…