之前没用过这个EventBus,然后看了一下EventBus的源码也没看明白,(-__-)b。反正大概就是弄一个优雅的方式实现了观察者模式吧。慢慢深入学习一下。

观察者模式其实就是生产者消费者的一个变种,就是一边有变化,然后有一个中介,也就是观察者去告诉消费者说:我说哥们啊,他们那边又变了,咱也跟着变吧!

然后观察者要么就是一个抽象类或者一个接口,里面有个update方法,需要每个处理的实例去实现,然后变化的那方持有这些实例,然后挨个去通知。

所以你也看到了,这个持有操作其实就是很不优雅的操作,所以我们用EventBus来看下这个地方到底是怎么实现的优雅的:

可以看到其实Guava中的EventBus的代码很少,只有几个类和注解。

注解有俩个:AllowConcurrentEvents和Subscribe,@AllowConcurrentEvents代表使用线程安全方式获得通知和@Subscribe代表这是一个订阅者,这俩注解和一起使用。

EventBus也就是观察者角色类有两个 EventBus和AsyncEventBus,前面是同步消息,后面支持异步消息。

DeadEvent是指没人关心的消息,可以做一下特殊处理,这个还是很有用,你可以获得有哪些消息根本没人消费过。

Subscriber订阅者对象,具体是哪个EventBus、哪个Listener、哪个方法 3者共同决定一个Subcriber

Dispatcher分发消息给上面那个(那些)Subscriber

SubscriberExceptionContext 订阅者抛出的异常上下文

SubscriberExceptionHandler 接口,处理订阅者抛出的异常

SubscriberRegistry 订阅动作,处理订阅者注册到一个EventBus的动作

我们来测试一下代码:

public class EventBusMain {
static final EventBus LINE_1St = new EventBus("first");
static AsyncEventBus LINE_2Ed = new AsyncEventBus("second", new Executor() {
@Override
public void execute(Runnable command) {
try {
Thread.sleep(10000L);
command.run();
} catch (InterruptedException e) {
System.out.println(e);
}
}
}); public static void main(String[] args) {
LINE_1St.register(new EventListener());
LINE_2Ed.register(new EventListener()); int cpuNums = Runtime.getRuntime().availableProcessors();
ExecutorService executorService = Executors.newFixedThreadPool(cpuNums * 1); Thread thread1 = new Thread() {
@Override
public void run() {
PromoEvent promoEvent = genPromoEvent();
LINE_1St.post(promoEvent);
}
};
Thread thread2 = new Thread() {
@Override
public void run() {
LINE_2Ed.post(genDimensionEvent());
}
}; Thread thread3 = new Thread() {
@Override
public void run() {
try {
Thread.sleep(5000);
LINE_1St.post(111);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
};
executorService.execute(thread1);
executorService.execute(thread2);
executorService.execute(thread3);
} static PromoEvent genPromoEvent() {
PromoEvent event = new PromoEvent(1, Lists.newArrayList(1L, 2L, 3L), System.currentTimeMillis(), 0L);
return event;
} static DimensionEvent genDimensionEvent() {
DimensionEvent event = new DimensionEvent(100, Lists.newArrayList(100L, 200L, 300L), System.currentTimeMillis(), 0L);
return event;
}

然后有两个自定义事件内容,基本上内容是一样的,只是用来测试:

public class PromoEvent {
private int activityId;
private List<Long> productIds;
private Long currentTime;
private Long triedTimes; /**
* @param activityId
* @param productIds
* @param currentTime
* @param triedTimes
*/
public PromoEvent(int activityId, List<Long> productIds, Long currentTime, Long triedTimes) {
super();
this.activityId = activityId;
this.productIds = productIds;
this.currentTime = currentTime;
this.triedTimes = triedTimes;
} public int getActivityId() {
return activityId;
} public void setActivityId(int activityId) {
this.activityId = activityId;
} public List<Long> getProductIds() {
return productIds;
} public void setProductIds(List<Long> productIds) {
this.productIds = productIds;
} public Long getCurrentTime() {
return currentTime;
} public void setCurrentTime(Long currentTime) {
this.currentTime = currentTime;
} public Long getTriedTimes() {
return triedTimes;
} public void setTriedTimes(Long triedTimes) {
this.triedTimes = triedTimes;
} @Override
public String toString() {
return "EventTest [activityId=" + activityId + ", productIds=" + productIds + ", currentTime=" + currentTime + ", triedTimes=" + triedTimes
+ "]";
}

事件监听:

public class EventListener {
private PromoEvent lastPromoMessage = null;
private DimensionEvent lastDimensionMessage = null;
private DeadEvent deadEvent = null; @Subscribe
public void listen(PromoEvent event) {
lastPromoMessage = event;
System.out.println("~~~~~~~~~~PromoEvent~~~~~~~~~~~~~");
System.out.println(event.toString());
} @Subscribe
public void listen(DimensionEvent event) {
lastDimensionMessage = event;
System.out.println("----------DimensionEvent---------");
System.out.println(event.toString());
} @Subscribe
public void listen(DeadEvent event) {
deadEvent = event;
System.out.println("===========DeadEvent=============");
System.out.println(event.toString());
} public PromoEvent getLastPromoMessage() {
return lastPromoMessage;
} public DimensionEvent getLastDimensionMessage() {
return lastDimensionMessage;
} public DeadEvent getDeadEvent() {
return deadEvent;

我们声明了两个Bus总线,用来分别存放消息体:

static final EventBus LINE_1St = new EventBus("first");
static AsyncEventBus LINE_2Ed = new AsyncEventBus("second", new Executor() {····});

然后声明3个线程,分别往同步总线里面放一个消息;延迟5秒往同步线程里面放一个无效消息;然后向延迟10秒的异步线程里面放一条消息;

然后监听器Listener分别处理这几个消息,下面是输出结果:

~~~~~~~~~~PromoEvent~~~~~~~~~~~~~
EventTest [activityId=1, productIds=[1, 2, 3], currentTime=1453096744555, triedTimes=0]
===========DeadEvent=============
com.google.common.eventbus.DeadEvent@4ffa5d1a
----------DimensionEvent---------
EventTest [activityId=100, productIds=[100, 200, 300], currentTime=1453096744555, triedTimes=0]

  

Guava包学习--EventBus的更多相关文章

  1. Guava包学习---Lists

    Guava包是我最近项目中同事推荐使用的,是google推出的库.里面的功能非常多,包括了集合.缓存.原生类型支持.并发库.通用注解.字符串处理.IO等.我们项目中使用到了guava依赖,但是实际上只 ...

  2. Guava包学习-Cache

    这段时间用到了ehcache和memcache,memcache只用来配置在tomcat中做负载均衡过程中的session共享,然后ehcache用来存放需要的程序中缓存. Guava中的Cache和 ...

  3. Guava包学习---Maps

    Maps包方法列表: 还是泛型创建Map: public static <K, V> HashMap<K, V> newHashMap() { return new HashM ...

  4. Guava包学习--Hash

    我们HashMap会有一个rehash的过程,为什么呢?因为java内建的散列码被限制为32位,而且没有分离散列算法和所作用的数据,所以替代算法比较难做.我们使用HashMap的时候它自身有一个reh ...

  5. Guava包学习---I/O

    Guava的I/O平时使用不太多,目前项目原因导致基本上只有在自己写一些文本处理小工具才用得到.但是I/O始终是程序猿最常遇到的需求和面试必问的知识点之一.同时Guava的I/O主要面向是时JDK5和 ...

  6. Guava包学习---Bimap

    Bimap也是Guava中提供的新集合类,别名叫做双向map,就是key->value,value->key,也就是你可以通过key定位value,也可以用value定位key. 这个场景 ...

  7. Guava包学习-Multimap

    它和上一章的MultiSet的继承结果很相似,只不过在上层的接口是Multimap不是Multiset. Multimap的特点其实就是可以包含有几个重复Key的value,你可以put进入多个不同v ...

  8. Guava包学习---Sets

    Sets包的内容和上一篇中的Lists没有什么大的区别,里面有些细节可以看一下: 开始的创建newHashSet()的各个重载方法.newConcurrentHashSet()的重载方法.newTre ...

  9. Guava包学习--Table

    Table,顾名思义,就好像HTML中的Table元素一样,其实就是行+列去确定的值,更准确的比喻其实就是一个二维矩阵. 其实它就是通过行+列两个key去找到一个value,然后它又containsv ...

随机推荐

  1. [转]Upgrading to Async with Entity Framework, MVC, OData AsyncEntitySetController, Kendo UI, Glimpse & Generic Unit of Work Repository Framework v2.0

    本文转自:http://www.tuicool.com/articles/BBVr6z Thanks to everyone for allowing us to give back to the . ...

  2. 跨域 cookies

    script标签请求的js脚本,如果跨域了,请求会带有外域的cookies信息. XMLHttpRequest请求跨域时,需要有Access-Control-*等的头信息,如果需要将cookies传输 ...

  3. .net下 本地锁、redis分布式锁、zk分布式锁的实现

    为什么要用锁? 大型站点在高并发的情况下,为了保持数据最终一致性就需要用到技术方案来支持.比如:分布式锁.分布式事务.有时候我们在为了保证某一个方法每次只能被一个调用者使用的时候,这时候我们也可以锁来 ...

  4. Java - 枚举与注解

    Enumeration 于Java 1.5增加的enum type... enum type是由一组固定的常量组成的类型,比如四个季节.扑克花色. 在出现enum type之前,通常用一组int常量表 ...

  5. Docker学习(四): 操作容器

    特别声明: 博文主要是学习过程中的知识整理,以便之后的查阅回顾.部分内容来源于网络(如有摘录未标注请指出).内容如有差错,也欢迎指正! =============系列文章============= 1 ...

  6. VMwear安装Centos7详细过程

      1.软硬件准备软件:推荐使用VMwear,我用的是VMwear 10镜像:CentOS7   硬件:因为是在宿主机上运行虚拟化软件安装centos,I3CPU双核.硬盘500G.内存4G以上. 2 ...

  7. linux ubuntu 学习总结(day01)基本命令学习

    个人总结,请尊重版权,转载请在显眼地方注明出处:https://www.cnblogs.com/sunshine5683/p/9948259.html day(01)基本命令学习 今天开始自学linu ...

  8. Python Django ORM基本增删改查

    工程下的urls.py中增加如下: from cmdb import views as cmdb #要把你要操作的项目import进来 urlpatterns = [ url(r'orm', cmdb ...

  9. 收藏的几个关于php面向对象教程

    面向对象的分析方法是利用面向对象的信息建模概念,如实体.关系.属性等,同时运用封装.继承.多态等机制来构造模拟现实系统的方法,学会了面向对象思想,能够大大提高php编程开发效率!本篇文章php中文网将 ...

  10. sql-(Cross||Outer)Apply

    Apply - 涉及以下两个步骤中的一步或两步(取决于Apply的类型): 1.A1:把右表表达式应用于左表的行 2.A2:添加外部行 Apply运算符把右表表达式应用于左输入的每一行.右表达式可以引 ...