观察者模式与Google Guava EventBus实现
概述
观察者模式又被称为发布-订阅(Publish/Subscribe)模式,属于行为型模式的一种。
它定义了一种一对多的依赖关系,让多个观察者对象同时监听某一个主题对象。这个主题对象在状态变化时,会通知所有的观察者对象,使他们能够自动更新自己。

简单实现
主题接口和观察者接口
//主题接口
public interface Subject {
//登记
void register(Observer observer);
//注销登记
void unregister(Observer observer);
//通知
void notify();
}
//观察者接口
public interface Observer {
//更新
void update();
}
实现类
//主题实现类
public class TeacherComeSubject implements Subject {
private List<Observer> observerList = new ArrayList<>();
@Override
public void register(Observer observer) {
observerList.add(observer);
}
@Override
public void unregister(Observer observer) {
observerList.remove(observer);
}
@Override
public void notfiy() {
for (Observer observer : observerList) {
observer.update();
}
}
}
//打游戏学生观察者
public class StudentPlayGameObserver implements Observer {
@Override
public void update() {
System.out.println("停止打游戏,马上写作业");
}
}
//聊天学生观察者
public class StudentChatObserver implements Observer {
@Override
public void update() {
System.out.println("停止聊天,马上写作业");
}
}
客户端
public static void main( String[] args ) {
Subject subject = new TeacherComeSubject();
Observer studentChatObserver = new StudentChatObserver();
Observer studentPlayGameObserver = new StudentPlayGameObserver();
subject.register(studentChatObserver);
subject.register(studentPlayGameObserver);
//老师来了
subject.notfiy();
}
执行结果:
停止聊天,马上写作业
停止打游戏,马上写作业
Google Guava 实现
EventBus是Guava中的事件处理机制,是观察者模式的优雅实现,使用起来也非常的简单。
它主要由三部分组成:事件总线、事件、事件监听
它的缺点是:只能单进程使用,项目异常重启或者退出不保证消息持久化。如果要支持分布式,还是要用MQ。

0. 引入依赖
<dependencies>
<dependency>
<groupId>com.google.guava</groupId>
<artifactId>guava</artifactId>
<version>20.0</version>
</dependency>
</dependencies>
1. 事件总线
//静态工具类
public class EventBusCenter{
private static final EventBus eventBus = new EventBus();
/**
* 登记事件监听
*
* @param eventListener
*/
public static void register(EventListener eventListener) {
eventBus.register(eventListener);
}
/**
* 注销登记事件监听
*
* @param eventListener
*/
public static void unregister(EventListener eventListener) {
eventBus.unregister(eventListener);
}
/**
* 推送事件
*
* @param eventListener
*/
public static void post(Event event) {
eventBus.post(event);
}
}
2. 事件
//事件接口
public interface Event {
String getName();
}
/**
* 老师来了事件
*/
public class TeacherComeEvent implements Event {
/**
* 事件名称
*/
private String name;
public TeacherComeEvent(String name) {
this.name = name;
}
@Override
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
/**
* 老师离开事件
*/
public class TeacherLeaveEvent implements Event {
private String name;
public TeacherLeaveEvent(String name) {
this.name = name;
}
@Override
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
3. 事件监听
//事件监听接口
public interface EventListener {
}
//老师事件监听,包括老师来了事件、老师走了事件
public class TeacherEventListener implements EventListener {
/**
* 监听老师来了事件
*
* @param event
*/
@Subscribe
public void teacherCome(TeacherComeEvent event) {
System.out.println(Thread.currentThread() + "-" + event.getName() + ":停止游戏,假装写作业");
}
/**
* 监听老师走了事件
*
* @param event
*/
@Subscribe
public void teacherLeave(TeacherLeaveEvent event) {
System.out.println(Thread.currentThread() + "-" + event.getName() + ":停止写作业,接着玩游戏");
}
}
使用
//事件监听器
EventListener eventListener = new TeacherEventListener();
//登记事件监听器
EventBusCenter.register(eventListener);
//推送事件
EventBusCenter.post(new TeacherComeEvent("数学老师来了"));
EventBusCenter.post(new TeacherLeaveEvent("数学老师走了"));
执行结果:
Thread[main,5,main]-数学老师来了:停止游戏,假装写作业
Thread[main,5,main]-数学老师来了:停止写作业,接着玩游戏
异步处理
比如说,事件发生之后,不想立即处理,而是异步处理,只需换一个 EventBus 即可
//修改EventBusCenter
//private static final EventBus eventBus = new EventBus();
//异步处理事件
private static final EventBus eventBus = new AsyncEventBus(Executors.newCachedThreadPool());
观察此时的执行结果:
Thread[pool-1-thread-1,5,main]-数学老师来了:停止游戏,假装写作业
Thread[pool-1-thread-2,5,main]-数学老师来了:停止写作业,接着玩游戏
观察者模式与Google Guava EventBus实现的更多相关文章
- 观察者模式与Guava EventBus
观察者模式 结构图 代码实现 public abstract class Subject { private List<Observer> observerList = new Array ...
- 设计模式:Observer(观察者)—— Guava EventBus
本文分为三个部分: Observer(观察者) Guava EventBus详解 Guava EventBus使用示例 1. Observer(观察者) 1.1 背景 我们设计系统时, ...
- Guava 12:Guava EventBus源码剖析
一.架构速读 传统上,Java的进程内事件分发都是通过发布者和订阅者之间的显式注册实现的.设计EventBus就是为了取代这种显示注册方式,使组件间有了更好的解耦.EventBus不是通用型的发布-订 ...
- Guava学习笔记:Google Guava 类库简介
http://www.cnblogs.com/peida/tag/Guava/ Guava 是一个 Google 的基于java1.6的类库集合的扩展项目,包括 collections, cachin ...
- Google Guava官方教程(中文版)
Google Guava官方教程(中文版) 原文链接 译文链接 译者: 沈义扬,罗立树,何一昕,武祖 校对:方腾飞 引言 Guava工程包含了若干被Google的 Java项目广泛依赖 的核心库, ...
- 初探Google Guava
Guava地址:https://github.com/google/guava 第一次接触我是在16年春github上,当时在找单机查缓存方法,google guava当初取名是因为JAVA的类库不好 ...
- Google Guava新手教程
以下资料整理自网络 一.Google Guava入门介绍 引言 Guavaproject包括了若干被Google的 Java项目广泛依赖 的核心库,比如:集合 [collections] . ...
- Google Guava 类库简介
Guava 是一个 Google开发的 基于java的类库集合的扩展项目,包括 collections, caching, primitives support, concurrency librar ...
- 开源介绍:Google Guava、Google Guice、Joda-Time
一.Guava 是一个 Google 的基于java1.6的类库集合的扩展项目,包括 collections, caching, primitives support, concurrency lib ...
随机推荐
- CF736D Permutations(伴随矩阵)
CF736D Permutations(伴随矩阵) Luogu 题解时间 首先把边直接放进邻接矩阵, 很明显行列式的奇偶和方案数的奇偶一样. 设 $ A_{ i , j } $ 为矩阵的该行列的余子式 ...
- 文字图片在wps中清晰化方法
在wps中双击图片出属性,然后再选择文字增强.选择对比增加即可.
- BIO、NIO、AIO 有什么区别?
BIO:Block IO 同步阻塞式 IO,就是我们平常使用的传统 IO,它的特点是模式简单使用方便,并发处理能力低.NIO:New IO 同步非阻塞 IO,是传统 IO 的升级,客户端和服务器端通过 ...
- Redis的集群搭建(四)
1.redis-cluster架构图 2.redis-cluster投票:容错 架构细节: (1)所有的redis节点彼此互联(PING-PONG机制),内部使用二进制协议优化传输速度和带宽. (2) ...
- Oracle入门基础(十二)一一储存过程及触发器
1.第一个存储过程 打印Hello World 调用存储过程: 1.exec sayhelloworld(); 2.begin sayhelloworld(); sayhelloworld(); en ...
- 说出 5 条 IO 的最佳实践?
IO 对 Java 应用的性能非常重要.理想情况下,你不应该在你应用的关键路径上 避免 IO 操作.下面是一些你应该遵循的 Java IO 最佳实践: a)使用有缓冲区的 IO 类,而不要单独读取字节 ...
- Tomcat配置文件之“server.xml”解析
<?xml version="1.0" encoding="UTF-8"?> <!--代表整个Servlet容器组件,是最顶层元素.它里面可以 ...
- pandas学习总结
什么是pandas pandas数据读取 03. Pandas数据结构 Pandas查询数据的几种方法
- 【uniapp 开发】uni-app 中如何打开外部应用
我们在开发 App 应用中,经常会遇到打开第三方程序的场景,比如打开手机淘宝.通过第三方浏览器打开一个 url 等等. App不像网页可以使用http超链接互相跳转,但手机os设计了scheme机制, ...
- css3,css的基础全局运用
浮动 1.浮动起来的盒子不占用位置,浮动了一个盒子下面的标准流的盒子会顶上来 可用清除浮动的方法来解决标准流会顶替位置的问题 清除浮动给父盒子加overflow: hidden; 鼠标经过事件 : ...