Guava学习笔记:EventBus
EventBus是Guava的事件处理机制,是设计模式中的观察者模式(生产/消费者编程模型)的优雅实现。对于事件监听和发布订阅模式,EventBus是一个非常优雅和简单解决方案,我们不用创建复杂的类和接口层次结构。
Observer模式是比较常用的设计模式之一,虽然有时候在具体代码里,它不一定叫这个名字,比如改头换面叫个Listener,但模式就是这个模式。手工实现一个Observer也不是多复杂的一件事,只是因为这个设计模式实在太常用了,Java就把它放到了JDK里面:Observable和Observer,从JDK 1.0里,它们就一直在那里。从某种程度上说,它简化了Observer模式的开发,至少我们不用再手工维护自己的Observer列表了。不过,如前所述,JDK里的Observer从1.0就在那里了,直到Java 7,它都没有什么改变,就连通知的参数还是Object类型。要知道,Java 5就已经泛型了。Java 5是一次大规模的语法调整,许多程序库从那开始重新设计了API,使其更简洁易用。当然,那些不做应对的程序库,多半也就过时了。这也就是这里要讨论知识更新的原因所在。今天,对于普通的应用,如果要使用Observer模式该如何做呢?答案是Guava的EventBus。
EventBus基本用法:
使用Guava之后, 如果要订阅消息, 就不用再继承指定的接口, 只需要在指定的方法上加上@Subscribe注解即可。代码如下:
消息封装类:
- public class TestEvent {
- private final int message;
- public TestEvent(int message) {
- this.message = message;
- System.out.println("event message:"+message);
- }
- public int getMessage() {
- return message;
- }
- }
消息接受类:
- public class EventListener {
- public int lastMessage = 0;
- @Subscribe
- public void listen(TestEvent event) {
- lastMessage = event.getMessage();
- System.out.println("Message:"+lastMessage);
- }
- public int getLastMessage() {
- return lastMessage;
- }
- }
测试类及输出结果:
- public class TestEventBus {
- @Test
- public void testReceiveEvent() throws Exception {
- EventBus eventBus = new EventBus("test");
- EventListener listener = new EventListener();
- eventBus.register(listener);
- eventBus.post(new TestEvent(200));
- eventBus.post(new TestEvent(300));
- eventBus.post(new TestEvent(400));
- System.out.println("LastMessage:"+listener.getLastMessage());
- ;
- }
- }
- //输出信息
- event message:200
- Message:200
- event message:300
- Message:300
- event message:400
- Message:400
- LastMessage:400
MultiListener的使用:
只需要在要订阅消息的方法上加上@Subscribe注解即可实现对多个消息的订阅,代码如下:
- public class MultipleListener {
- public Integer lastInteger;
- public Long lastLong;
- @Subscribe
- public void listenInteger(Integer event) {
- lastInteger = event;
- System.out.println("event Integer:"+lastInteger);
- }
- @Subscribe
- public void listenLong(Long event) {
- lastLong = event;
- System.out.println("event Long:"+lastLong);
- }
- public Integer getLastInteger() {
- return lastInteger;
- }
- public Long getLastLong() {
- return lastLong;
- }
- }
测试类:
- public class TestMultipleEvents {
- @Test
- public void testMultipleEvents() throws Exception {
- EventBus eventBus = new EventBus("test");
- MultipleListener multiListener = new MultipleListener();
- eventBus.register(multiListener);
- eventBus.post(new Integer(100));
- eventBus.post(new Integer(200));
- eventBus.post(new Integer(300));
- eventBus.post(new Long(800));
- eventBus.post(new Long(800990));
- eventBus.post(new Long(800882934));
- System.out.println("LastInteger:"+multiListener.getLastInteger());
- System.out.println("LastLong:"+multiListener.getLastLong());
- }
- }
- //输出信息
- event Integer:100
- event Integer:200
- event Integer:300
- event Long:800
- event Long:800990
- event Long:800882934
- LastInteger:300
- LastLong:800882934
Dead Event:
如果EventBus发送的消息都不是订阅者关心的称之为Dead Event。实例如下:
- public class DeadEventListener {
- boolean notDelivered = false;
- @Subscribe
- public void listen(DeadEvent event) {
- notDelivered = true;
- }
- public boolean isNotDelivered() {
- return notDelivered;
- }
- }
测试类:
- public class TestDeadEventListeners {
- @Test
- public void testDeadEventListeners() throws Exception {
- EventBus eventBus = new EventBus("test");
- DeadEventListener deadEventListener = new DeadEventListener();
- eventBus.register(deadEventListener);
- eventBus.post(new TestEvent(200));
- eventBus.post(new TestEvent(300));
- System.out.println("deadEvent:"+deadEventListener.isNotDelivered());
- }
- }
- //输出信息
- event message:200
- event message:300
- deadEvent:true
说明:如果没有消息订阅者监听消息, EventBus将发送DeadEvent消息,这时我们可以通过log的方式来记录这种状态。
Event的继承:
如果Listener A监听Event A, 而Event A有一个子类Event B, 此时Listener A将同时接收Event A和B消息,实例如下:
Listener 类:
- public class NumberListener {
- private Number lastMessage;
- @Subscribe
- public void listen(Number integer) {
- lastMessage = integer;
- System.out.println("Message:"+lastMessage);
- }
- public Number getLastMessage() {
- return lastMessage;
- }
- }
- public class IntegerListener {
- private Integer lastMessage;
- @Subscribe
- public void listen(Integer integer) {
- lastMessage = integer;
- System.out.println("Message:"+lastMessage);
- }
- public Integer getLastMessage() {
- return lastMessage;
- }
- }
测试类:
- public class TestEventsFromSubclass {
- @Test
- public void testEventsFromSubclass() throws Exception {
- EventBus eventBus = new EventBus("test");
- IntegerListener integerListener = new IntegerListener();
- NumberListener numberListener = new NumberListener();
- eventBus.register(integerListener);
- eventBus.register(numberListener);
- eventBus.post(new Integer(100));
- System.out.println("integerListener message:"+integerListener.getLastMessage());
- System.out.println("numberListener message:"+numberListener.getLastMessage());
- eventBus.post(new Long(200L));
- System.out.println("integerListener message:"+integerListener.getLastMessage());
- System.out.println("numberListener message:"+numberListener.getLastMessage());
- }
- }
- //输出类
- Message:100
- Message:100
- integerListener message:100
- numberListener message:100
- Message:200
- integerListener message:100
- numberListener message:200
说明:在这个方法中,我们看到第一个事件(新的整数(100))是收到两个听众,但第二个(新长(200 l))只能到达NumberListener作为整数一不是创建这种类型的事件。可以使用此功能来创建更通用的监听器监听一个广泛的事件和更详细的具体的特殊的事件。
一个综合实例:
- public class UserThread extends Thread {
- private Socket connection;
- private EventBus channel;
- private BufferedReader in;
- private PrintWriter out;
- public UserThread(Socket connection, EventBus channel) {
- this.connection = connection;
- this.channel = channel;
- try {
- in = new BufferedReader(new InputStreamReader(connection.getInputStream()));
- out = new PrintWriter(connection.getOutputStream(), true);
- } catch (IOException e) {
- e.printStackTrace();
- System.exit(1);
- }
- }
- @Subscribe
- public void recieveMessage(String message) {
- if (out != null) {
- out.println(message);
- System.out.println("recieveMessage:"+message);
- }
- }
- @Override
- public void run() {
- try {
- String input;
- while ((input = in.readLine()) != null) {
- channel.post(input);
- }
- } catch (IOException e) {
- e.printStackTrace();
- }
- //reached eof
- channel.unregister(this);
- try {
- connection.close();
- } catch (IOException e) {
- e.printStackTrace();
- }
- in = null;
- out = null;
- }
- }
- mport java.io.IOException;
- import java.net.ServerSocket;
- import java.net.Socket;
- import com.google.common.eventbus.EventBus;
- public class EventBusChat {
- public static void main(String[] args) {
- EventBus channel = new EventBus();
- ServerSocket socket;
- try {
- socket = new ServerSocket(4444);
- while (true) {
- Socket connection = socket.accept();
- UserThread newUser = new UserThread(connection, channel);
- channel.register(newUser);
- newUser.start();
- }
- } catch (IOException e) {
- e.printStackTrace();
- }
- }
- }
说明:用telnet命令登录:telnet 127.0.0.1 4444 ,如果你连接多个实例你会看到任何消息发送被传送到其他实例。
Guava学习笔记:EventBus的更多相关文章
- Guava学习笔记目录
Guava 是一个 Google 的基于java1.6的类库集合的扩展项目,包括 collections, caching, primitives support, concurrency libra ...
- guava 学习笔记 使用瓜娃(guava)的选择和预判断使代码变得简洁
guava 学习笔记 使用瓜娃(guava)的选择和预判断使代码变得简洁 1,本文翻译自 http://eclipsesource.com/blogs/2012/06/06/cleaner-code- ...
- guava 学习笔记(二) 瓜娃(guava)的API快速熟悉使用
guava 学习笔记(二) 瓜娃(guava)的API快速熟悉使用 1,大纲 让我们来熟悉瓜娃,并体验下它的一些API,分成如下几个部分: Introduction Guava Collection ...
- Guava学习笔记:EventBus(转)
EventBus是Guava的事件处理机制,是设计模式中的观察者模式(生产/消费者编程模型)的优雅实现.对于事件监听和发布订阅模式,EventBus是一个非常优雅和简单解决方案,我们不用创建复杂的类和 ...
- Guava学习笔记(一)概览
Guava是谷歌开源的一套Java开发类库,以简洁的编程风格著称,提供了很多实用的工具类, 在之前的工作中应用过Collections API和Guava提供的Cache,不过对Guava没有一个系统 ...
- Guava学习笔记:Google Guava 类库简介
http://www.cnblogs.com/peida/tag/Guava/ Guava 是一个 Google 的基于java1.6的类库集合的扩展项目,包括 collections, cachin ...
- Guava学习笔记(1):Optional优雅的使用null
转自:http://www.cnblogs.com/peida/archive/2013/06/14/Guava_Optional.html 参考:[Google Guava] 1.1-使用和避免nu ...
- Guava学习笔记:Optional优雅的使用null
在我们学习和使用Guava的Optional之前,我们需要来了解一下Java中null.因为,只有我们深入的了解了null的相关知识,我们才能更加深入体会领悟到Guava的Optional设计和使用上 ...
- Guava学习之EventBus
一.EventBus的使用案例 EventBus是Guava的事件处理机制,是设计模式中的观察者模式(生产/消费者编程模型)的优雅实现.对于事件监听和发布订阅模式,EventBus是一个非常优雅和简单 ...
随机推荐
- 移动端IM开发需要面对的技术问题
1.前言 这两年多一直从事网易云信 iOS 端 IM SDK的开发,期间不断有兄弟部门的同事和合作伙伴过来问各种技术细节,干脆统一介绍下一个IM APP的方方面面,包括技术选型(包括通讯方式,网络连接 ...
- JS原生第二篇 (帅哥)
1.1 Javascript 作用 1. 网页特效 2. 用户交互 3. 表单验证 Js 就是可以用来控制 结构 和 样式 . 1.2 体验js 认识常用的三个输出语句. 都属于 ...
- [转载]基于TFS实践敏捷-实现用户场景
您是新用户的 Visual Studio 应用程序生命周期管理 (ALM) 和 Team Foundation Server (TFS) 吗? 您想知道如何您和您的团队可以获得最大受益的这些工具来生成 ...
- NodeJS实战:Express+Mongoose+ejs
元宵还没到,先向所有朋友拜一个晚年~~~ 文章目录: 1.组件版本号 -- --node -- --express -- --Mongoose 2.初始化项目 firstblood -- --用 ex ...
- 如何给wordpress首页自动显示文章内容的第一个图片
敏捷个人手机应用中使用到的数据来源于wordpress中,因为自己写的页面,所以可以自己写代码获取文章内容的第一个图片作为文章缩略图来显示,这样用户看到首页时图文并茂,感觉会好一些. 现在后台简单的使 ...
- Android自动化测试之Monkeyrunner学习笔记(一)
Android自动化测试之Monkeyrunner学习笔记(一) 因项目需要,开始研究Android自动化测试方法,对其中的一些工具.方法和框架做了一些简单的整理,其中包括Monkey.Monkeyr ...
- Theano入门神经网络(二) 实现一个XOR门
与非门的图片如下 示意图 详细解释: 1 定义变量的代码,包括了输入.权值.输出等.其中激活函数采用的是sigmod函数 # -*- coding: utf-8 -*- __author__ = 'A ...
- 【转载】ASP.NET MVC Web API 的路由选择
此文章描述了ASP.NET Web API如何将Http请求路由到controller. 路由表 在ASP.NET Web API中,controller是用来处理HTTP请求的一个类.这个类中用于处 ...
- 反射(Reflection)的SetValue遇上DBNULL转换为string
有网友回馈说提供的代码有bug.你可以从这个链接下载得到:http://www.cnblogs.com/insus/p/3384472.html 其中有一个Utility,它是把DataTable转换 ...
- HoverTree开源项目已经实现管理员登录
ASP.NET开源项目HoverTree已经实现了管理员登录功能,最新代码请到以下网址查看.http://hovertree.com/down/ 点击Clone右边的Download就可以下载最新开发 ...