EventBus 事件总线 案例
简介
EventBus是一个【发布 / 订阅】的事件总线。简单点说,就是两人【约定】好怎么通信,一人发布消息,另外一个约定好的人立马接收到你发的消息。EventBus是一款针对Android优化的发布/订阅事件总线。主要功能是替代Intent,Handler,BroadCast在Fragment,Activity,Service,线程之间传递消息。优点:开销小,代码更优雅,将发送者和接收者解耦。用处:相信大家都用过【Handle】了进行线程通信,回调方法进行通信,是不是觉得特麻烦。EventBus就可以帮减少很多事,不管你在【任何地方】发布一个事件,接收者都能立马接收到你的消息,不用你考虑android【子线程操作UI线程】的问题!EvenBus简化了应用程序内【各组件间】、【组件与后台线程间】的通信。它的效果和Handler的效果大致相同,但是实现原理和使用方法是完全不同的,它是基于【保存】相应方法,然后通过【反射】机制来实现的。包含4个成分:发布者,订阅者,事件,总线。关系:订阅者订阅事件到总线,发送者发布事件;订阅者可以订阅多个事件,发送者可以发布任何事件,发布者同时也可以是订阅者。注册订阅者:EventBus.getDefault().register(this);这个方法通常在onCreate方法中进行注册。解绑订阅者:EventBus.getDefault().unregister(this);这个方法通常在onDestroy方法中进行解绑。发布者是不需要注册的,只有订阅者才需要注册。若收不到事件,比如在Activity的oncreat中发送事件,在Fragment中的oncreat中接收事件,可能是因为发送事件时,接收事件的Fragment还未注册EventBus。这种情况可以采用发送Sticky Event(粘性事件)来解决这样注册之前发送的sticky事件的最近的一个会保存在内存中,错过这个事件的发送的情况下,也可以通过getStickyEvent收到。注册时使用:EventBus.getDefault().registerSticky(this);// 实际中发现不以Sticky的形式注册也完全可以收到Sticky事件获取时使用:StickyEventBusBeans bean = EventBus.getDefault().getStickyEvent(StickyEventBusBeans.class);还可以移除:EventBus.getDefault().removeStickyEvent(bean);约定的收到事件要执行的方法要求:方法名必须以【onEvent】开头,必须为非静态、【public】权限、有且仅有一个参数
- onEvent:如果使用onEvent作为订阅函数,那么该事件在哪个线程发布出来的,onEvent就会在这个线程中运行,也就是说发布事件和接收事件线程在同一个线程。使用这个方法时,在onEvent方法中不能执行耗时操作,如果执行耗时操作容易导致事件分发延迟。
- onEventMainThread:如果使用onEventMainThread作为订阅函数,那么不论事件是在哪个线程中发布出来的,onEventMainThread都会在UI线程中执行,接收事件就会在UI线程中运行,这个在Android中是非常有用的,因为在Android中只能在UI线程中更新UI,所以在onEvnetMainThread方法中是不能执行耗时操作的。
- onEventBackground:如果使用onEventBackgrond作为订阅函数,那么如果事件是在UI线程中发布出来的,那么onEventBackground就会在子线程中运行,如果事件本来就是子线程中发布出来的,那么onEventBackground函数直接在该子线程中执行。
- onEventAsync:使用这个函数作为订阅函数,那么无论事件在哪个线程发布,都会创建新的子线程在执行onEventAsync.
注意post方法的参数,EventBus是根据这四个方法的参数来决定哪个类接收事件的,发布者的参数和某个订阅者这四个方法的参数一样,则执行这个订阅者的这个方法。当订阅者收到事件后,就会自动执行上面这四个方法,如果写了多个,则都会执行。EventBus带来的好处和引入的问题好处比较明显,就是独立出一个发布订阅模块,调用者可以通过使用这个模块,屏蔽一些线程切换问题,简单地实现发布订阅功能。坏处可能比较隐晦,但这些需要足够引起我们的重视
- 大量的滥用,将导致逻辑的分散,出现问题后很难定位
- 没办法实现强类型,在编译的时候就发现问题,(Otto实现了这个,但性能有问题)
- 在实现上通过一个很弱的协议,比如onEventXXX(XXX表示ThreadModel),来实现线程的切换,代码可读性有些问题,IDE无法识别这些协议,对IDE不友好
EventBus 、 Otto 、 BroadcastReceiver三者比较1、单从使用上看,EventBus > Otto > BroadcastReceiver,当然BroadcastReceiver作为系统内置组件,有一些前两者没有的功能2、EventBus最简洁,Otto最符合Guava EventBus的设计思路, BroadcastReceiver最难使用。3、Otto使用注解定义订阅/发布者的角色,@Subscribe为订阅者,@Produce为发布者,方法名称可以自定义。EventBus规定onEvent方法固定作为订阅者接受事件的方法,应该是参考了"约定优于配置"的思想。4、Otto为了性能,代码意图清晰,要求@Subscribe,@Produce方法必须定义在直接的作用类上,而不能定义在基类而被继承。5、Otto要求发布者也需要register和unregister,而EventBus的发布者是不需要的。
MainActivity
public class MainActivity extends Activity {TextView textView;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);EventBus.getDefault().register(this);//在onCreate里面进行事件的订阅,onDestroy里面进行事件的取消textView = new TextView(this);textView.setText("点击进入第二个Activity");textView.setTextSize(TypedValue.COMPLEX_UNIT_SP, 18);textView.setGravity(Gravity.CENTER);textView.setOnClickListener(new OnClickListener() {@Overridepublic void onClick(View v) {startActivity(new Intent(MainActivity.this, ItemActivity.class));EventBus.getDefault().postSticky(new StickyEventBusBeans(StickyEventBusBeans.TIME, "来自MainActivity\n" + new SimpleDateFormat("HH:mm:ss").format(new Date())));}});setContentView(textView);}@Overrideprotected void onDestroy() {super.onDestroy();EventBus.getDefault().unregister(this);}//******************************************************************************************/*** 方法名必须以【onEvent】开头,必须为非静态的、public权限的、有且仅有一个参数* 【MainThread】意思这个方法最终要在UI线程执行(不管在哪个线程发布的消息);当指定事件发布的时候,这个方法就会在UI线程自动执行*/public void onEventMainThread(StringEventBusBean event) {textView.setText(event.getContent());}/*** 以onEvent作为订阅函数,那么该事件在哪个线程发布出来的,onEvent就会在哪个线程中运行*/public void onEvent(NormalEventBusBeans event) {switch (event.getFlag()) {case NormalEventBusBeans.LOGIN://主线程发布的事件textView.setText("登录成功\n" + Thread.currentThread().getName());break;case NormalEventBusBeans.LOGOUT://子线程发布的事件try {Log.i("bqt", Thread.currentThread().getName());Thread.sleep(20 * 1000);//这是在子线程中Log.i("bqt", "如果不是在子线程中,肯定会直接卡死(ANR)");} catch (InterruptedException e) {e.printStackTrace();}break;default:break;}}}
第二个Activity
public class ItemActivity extends Activity {@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_item);}}<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"android:layout_width="match_parent"android:layout_height="match_parent"android:orientation="horizontal" ><fragmentandroid:id="@+id/item_list"android:name="com.bqt.eventbus.ItemListFragment"android:layout_width="0dip"android:layout_height="match_parent"android:layout_weight="1" /><fragmentandroid:id="@+id/item_detail_container"android:name="com.bqt.eventbus.ItemDetailFragment"android:layout_width="0dip"android:layout_height="match_parent"android:layout_weight="1" /></LinearLayout>
列表Fragment
public class ItemListFragment extends ListFragment {private List<String> items = new ArrayList<String>();@Overridepublic void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);EventBus.getDefault().register(this);}@Overridepublic void onDestroy() {super.onDestroy();EventBus.getDefault().unregister(this);}//******************************************************************************************@Overridepublic void onViewCreated(View view, Bundle savedInstanceState) {super.onViewCreated(view, savedInstanceState);// 开启线程加载列表new Thread() {public void run() {try {items.add("在UI线程中发布一个StringEventBusBean事件,通知右侧的fragment和MainActivity刷新数据");items.add("在UI线程中发布一个NormalEventBusBeans事件,通知MainActivity已登录");items.add("在子线程中发布一个NormalEventBusBeans事件,通知MainActivity已注销");items.add("在子线程中发布一个StringEventBusBean事件,通知右侧的fragment和MainActivity刷新数据");items.add("发布一个StickyEventBusBeans事件,此事件会把之前的覆盖掉");items.add("获取此Sticky事件,并通过一个StringEventBusBean事件把其中的内容转发出去");items.add("移除此Sticky事件\n(粘性事件)");Thread.sleep(300); // 模拟延时EventBus.getDefault().post(new ListEventBusBean(items));// 在【后台线程】发布一个参数为ListEventBusBean的事件} catch (InterruptedException e) {e.printStackTrace();}};}.start();}@Override//【This method will be called when an item in the list is selected】【position The position of the view in the list】【id The row id of the item that was clicked】public void onListItemClick(ListView listView, View view, int position, long id) {super.onListItemClick(listView, view, position, id);switch (position) {case 0:EventBus.getDefault().post(new StringEventBusBean("我是\nUI线程的\n包青天"));break;case 1:EventBus.getDefault().post(new NormalEventBusBeans(NormalEventBusBeans.LOGIN));getActivity().finish();break;case 2:new Thread(new Runnable() {@Overridepublic void run() {EventBus.getDefault().post(new NormalEventBusBeans(NormalEventBusBeans.LOGOUT));}}).start();getActivity().finish();break;case 3:new Thread(new Runnable() {@Overridepublic void run() {EventBus.getDefault().post(new StringEventBusBean("我是\n子线程的\n包青天"));}}).start();break;case 4:EventBus.getDefault().postSticky(new StickyEventBusBeans(StickyEventBusBeans.TIME, "来自ItemListFragment\n" + new SimpleDateFormat("HH:mm:ss").format(new Date())));break;case 5:StickyEventBusBeans bean5 = EventBus.getDefault().getStickyEvent(StickyEventBusBeans.class);if (bean5 != null) EventBus.getDefault().post(new StringEventBusBean(bean5.getContent()));else Toast.makeText(getActivity().getApplicationContext(), "不存在粘性事件哦", Toast.LENGTH_SHORT).show();break;case 6:StickyEventBusBeans bean6 = EventBus.getDefault().getStickyEvent(StickyEventBusBeans.class);if (bean6 != null) {boolean success = EventBus.getDefault().removeStickyEvent(bean6);//还可以移除所有粘性事件if (success) Toast.makeText(getActivity().getApplicationContext(), "移除成功", Toast.LENGTH_SHORT).show();} else Toast.makeText(getActivity().getApplicationContext(), "不存在粘性事件哦", Toast.LENGTH_SHORT).show();break;}}public void onEventMainThread(ListEventBusBean event) {ListAdapter adapter = new ArrayAdapter<String>(getActivity(), R.layout.item, R.id.tv_name, event.getItems());setListAdapter(adapter);}}
详情Fragment
public class ItemDetailFragment extends Fragment {private TextView tv_info;@Overridepublic void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);EventBus.getDefault().register(this);// 看来扯蛋的事情不止一丢丢:不以Sticky的形式注册也完全可以收到Sticky事件}@Overridepublic void onDestroy() {super.onDestroy();EventBus.getDefault().unregister(this);}@Overridepublic View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {tv_info = new TextView(getActivity());tv_info.setTextColor(Color.BLUE);tv_info.setTextSize(TypedValue.COMPLEX_UNIT_SP, 18);tv_info.setBackgroundColor(0x3300ff00);tv_info.setGravity(Gravity.CENTER);onGetStickyEvent();//此广播要自己手动获取(其实只是把此事件保存了起来,并不会主动通知)return tv_info;}//******************************************************************************************/** List点击时会发送此事件,接收到事件后更新详情 */public void onEventMainThread(StringEventBusBean item) {if (item != null) tv_info.setText(item.getContent());}/**这只是供本地调用的普通方法,对其声明没有任何要求*/private void onGetStickyEvent() {StickyEventBusBeans bean = EventBus.getDefault().getStickyEvent(StickyEventBusBeans.class);if (bean == null) return;switch (bean.getFlag()) {case StickyEventBusBeans.TIME://这个消息是从MainActivity传过来的,消息发送时ItemDetailFragment还没创建,但是可以用postSticky方式发送的消息仍然可以在这里获取tv_info.setText(bean.getContent());break;default:break;}}}
事件bean
/*** 一个自定义的类,用于通知,不需要传递数据*/public class NormalEventBusBeans {/**事件标志 */private int flag;public NormalEventBusBeans(int flag) {this.flag = flag;}public int getFlag() {return flag;}/**退出登录 */public static final int LOGOUT = 1002;/**登录*/public static final int LOGIN = 1001;}/*** 一个自定义的类,用于传递一个String对象*/public class StringEventBusBean {private String content;public StringEventBusBean(String content) {this.content = content;}public String getContent() {return content;}}/*** 一个自定义的类,用于传递一个List<String>集合*/public class ListEventBusBean {private List<String> items;public ListEventBusBean(List<String> items) {this.items = items;}public List<String> getItems() {return items;}}/*** 一个自定义的类,用于Sticky类型的通知,可以传两个参数*/public class StickyEventBusBeans {/**事件标志 */private int flag;/**事件内容 */private String content;public StickyEventBusBeans(int flag, String content) {this.content = content;this.flag=flag;}public int getFlag() {return flag;}public String getContent() {return content;}public static final int TIME = 10086;}
附件列表
EventBus 事件总线 案例的更多相关文章
- Guava - EventBus(事件总线)
Guava在guava-libraries中为我们提供了事件总线EventBus库,它是事件发布订阅模式的实现,让我们能在领域驱动设计(DDD)中以事件的弱引用本质对我们的模块和领域边界很好的解耦设计 ...
- EventBus(事件总线)
EventBus(事件总线) Guava在guava-libraries中为我们提供了事件总线EventBus库,它是事件发布订阅模式的实现,让我们能在领域驱动设计(DDD)中以事件的弱引用本质对我们 ...
- dhroid - eventbus 事件总线
你听过onClick 事件,onItemClick 事件,事件总线不一定听过吧, eventbus 事件总线也是一个编程思想,为什么要设计EventBus了,因为他是领域驱动设计中比不可少的模块,它承 ...
- EventBus事件总线
EventBus事件总线的使用-自己实现事件总线 在C#中,我们可以在一个类中定义自己的事件,而其他的类可以订阅该事件,当某些事情发生时,可以通知到该类.这对于桌面应用或者独立的windows服务 ...
- C#总结(六)EventBus事件总线的使用-自己实现事件总线
在C#中,我们可以在一个类中定义自己的事件,而其他的类可以订阅该事件,当某些事情发生时,可以通知到该类.这对于桌面应用或者独立的windows服务来说是非常有用的.但对于一个web应用来说是有点问题的 ...
- Android 开发 框架系列 EventBus 事件总线
介绍 GitHub:https://github.com/greenrobot/EventBus 先聊聊EventBus 线程总线是干什么的,使用环境,优点.缺点. 干什么的? 一句话,简单统一数据传 ...
- EventBus事件总线(牛x版)
事件总线: public interface IEventBus { void Trigger<TEvent>(TEvent eventData, string topic = null) ...
- EventBus 事件总线之我的理解
用例:假设公司发布了一个公告 需要通过短信 和 邮件分别2种方式 通知员工 1:首先我们建立领域模型 /// <summary> /// 领域核心基类 /// </summary&g ...
- Orchard EventBus 事件总线及 IEventHandler作用
事件总线接口定义: public interface IEventBus : IDependency { IEnumerable Notify(string messageName, IDiction ...
随机推荐
- 【USACO 1.3.1】混合牛奶
[题目描述] 由于乳制品产业利润很低,所以降低原材料(牛奶)价格就变得十分重要.帮助梅丽乳业找到最优的牛奶采购方案. 梅丽乳业从一些奶农手中采购牛奶,并且每一位奶农为乳制品加工企业提供的价格是不同的. ...
- ERROR:the server has either erred or is incapable of performing the requested operation
openstack中,有时会经常出现这种错误,原因无二,一是安全组没有设置正确,二是openstack中网络配置会有些问题或者是相关的服务没有启动. 解决方法:1.安全组问题在nova.conf和ne ...
- 【转载】ASP.NET页面运行机制以及请求处理流程
本文转至 ASP.NET页面运行机制以及请求处理流程 IIS处理页面的运行机制 IIS自身是不能处理像ASPX扩展名这样的页面,只能直接请求像HTML这样的静态文件,之所以能处理ASPX这样扩展名的页 ...
- 获取当前元素节点的position和宽高(兼容)
function objxy(n){ var o=document.getElementById(n),x=0,y=0,w=o.offsetWidth,h=o.offsetHeight if(o.ge ...
- 转发:[Python]内存管理
本文为转发,原地址为:http://chenrudan.github.io/blog/2016/04/23/pythonmemorycontrol.html 本文主要为了解释清楚python的内存管理 ...
- Python hashlib模块 (主要记录md5加密)
python提供了一个进行hash加密的模块:hashlib 下面主要记录下其中的md5加密方式(sha1加密一样把MD5换成sha1) >>> import hashlib > ...
- decimal类型数据如何保留两位小数
日常开发中,decimal作为货币类型,经常会处理保留两位小数的问题. 本站整理两种,decimal类型保留两位小数的方法. 第一种: decimal d = 46.28111m;string res ...
- 『在线工具』 基于 xsser.me 源码 + BootStrap 前端 的 XSS 平台
乌云社区上一个小伙伴的对xsser.me 的源码做了 BS 的优化,本人已经搭建好,提供给大家免费使用,大牛求绕过,多谢. 地址: http://xss.evilclay.com (目前开放注册,不需 ...
- CONTEXT MENU简介
安卓中的上下文菜单是通过长按控件元素触发的,要注意的是每次都会触发onCreateContextMenu方法: main.xml <?xml version="1.0" en ...
- XML PULL模型
student.xml <?xml version="1.0" encoding="utf-8"?> <stundets> <st ...