EventBus使用详解(二)

一、概述

前一篇给大家装简单演示了EventBus的onEventMainThread()函数的接收,其实EventBus还有另外有个不同的函数,他们分别是:

1、onEvent

2、onEventMainThread

3、onEventBackgroundThread

4、onEventAsync

这四种订阅函数都是使用onEvent开头的,它们的功能稍有不同,在介绍不同之前先介绍两个概念:

告知观察者事件发生时通过EventBus.post函数实现,这个过程叫做事件的发布,观察者被告知事件发生叫做事件的接收,是通过下面的订阅函数实现的。

onEvent:如果使用onEvent作为订阅函数,那么该事件在哪个线程发布出来的,onEvent就会在这个线程中运行,也就是说发布事件和接收事件线程在同一个线程。使用这个方法时,在onEvent方法中不能执行耗时操作,如果执行耗时操作容易导致事件分发延迟。

onEventMainThread:如果使用onEventMainThread作为订阅函数,那么不论事件是在哪个线程中发布出来的,onEventMainThread都会在UI线程中执行,接收事件就会在UI线程中运行,这个在Android中是非常有用的,因为在Android中只能在UI线程中跟新UI,所以在onEvnetMainThread方法中是不能执行耗时操作的。

onEventBackground:如果使用onEventBackgrond作为订阅函数,那么如果事件是在UI线程中发布出来的,那么onEventBackground就会在子线程中运行,如果事件本来就是子线程中发布出来的,那么onEventBackground函数直接在该子线程中执行。

onEventAsync:使用这个函数作为订阅函数,那么无论事件在哪个线程发布,都会创建新的子线程在执行onEventAsync.

二、实战

1、解析

上面列出的这四个函数,关键问题在于,我们怎么指定调用哪个函数呢?

我们先研究一下,上一篇中是怎么调用的onEventMainThread函数,除了在接收端注册与反注册以后,关键问题在于新建的一个类:

新建一个类:

[java] view
plain
 copy

  1. package com.harvic.other;
  2. public class FirstEvent {
  3. private String mMsg;
  4. public FirstEvent(String msg) {
  5. // TODO Auto-generated constructor stub
  6. mMsg = msg;
  7. }
  8. public String getMsg(){
  9. return mMsg;
  10. }
  11. }

发送时:

[java] view
plain
 copy

  1. EventBus.getDefault().post(new FirstEvent("FirstEvent btn clicked"));

接收时:

[java] view
plain
 copy

  1. public void onEventMainThread(FirstEvent event) {
  2. ……
  3. }

发现什么问题了没?

没错,发送时发送的是这个类的实例,接收时参数就是这个类实例。

所以!!!!!!当发过来一个消息的时候,EventBus怎么知道要调哪个函数呢,就看哪个函数传进去的参数是这个类的实例,哪个是就调哪个。那如果有两个是呢,那两个都会被调用!!!!

为了证明这个问题,下面写个例子,先看下效果

2、实例

先看看我们要实现的效果:

这次我们在上一篇的基础上,新建三个类:FirstEvent、SecondEvent、ThirdEvent,在第二个Activity中发送请求,在MainActivity中接收这三个类的实例,接收时的代码为:

[java] view
plain
 copy

  1. public void onEventMainThread(FirstEvent event) {
  2. Log.d("harvic", "onEventMainThread收到了消息:" + event.getMsg());
  3. }
  4. public void onEventMainThread(SecondEvent event) {
  5. Log.d("harvic", "onEventMainThread收到了消息:" + event.getMsg());
  6. }
  7. public void onEvent(ThirdEvent event) {
  8. Log.d("harvic", "OnEvent收到了消息:" + event.getMsg());
  9. }

使用两个onEventMainThread分别接收FirstEvent实例的消息和SecondEvent实例的消息,使用onEvent接收ThirdEvent实例的消息。界面操作及结果如下:

Log输出结果:

可以看到,在发送FirstEvent时,在MainActiviy中虽然有三个函数,但只有第一个onEventMainThread函数的接收参数是FirstEvent,所以会传到它这来接收。所以这里识别调用EventBus中四个函数中哪个函数,是通过参数中的实例来决定的。

因为我们是在上一篇例子的基础上完成的,所以这里的代码就不详细写了,只写改动的部分。

1、三个类

[java] view
plain
 copy

  1. package com.harvic.other;
  2. public class FirstEvent {
  3. private String mMsg;
  4. public FirstEvent(String msg) {
  5. // TODO Auto-generated constructor stub
  6. mMsg = msg;
  7. }
  8. public String getMsg(){
  9. return mMsg;
  10. }
  11. }

[java] view
plain
 copy

  1. package com.harvic.other;
  2. public class SecondEvent{
  3. private String mMsg;
  4. public SecondEvent(String msg) {
  5. // TODO Auto-generated constructor stub
  6. mMsg = "MainEvent:"+msg;
  7. }
  8. public String getMsg(){
  9. return mMsg;
  10. }
  11. }

[java] view
plain
 copy

  1. package com.harvic.other;
  2. public class ThirdEvent {
  3. private String mMsg;
  4. public ThirdEvent(String msg) {
  5. // TODO Auto-generated constructor stub
  6. mMsg = msg;
  7. }
  8. public String getMsg(){
  9. return mMsg;
  10. }
  11. }

2、发送

然后在SecondActivity中新建三个按钮,分别发送不同的类的实例,代码如下:

[java] view
plain
 copy

  1. package com.harvic.tryeventbus2;
  2. import com.harvic.other.FirstEvent;
  3. import com.harvic.other.SecondEvent;
  4. import com.harvic.other.ThirdEvent;
  5. import de.greenrobot.event.EventBus;
  6. import android.app.Activity;
  7. import android.os.Bundle;
  8. import android.view.View;
  9. import android.widget.Button;
  10. public class SecondActivity extends Activity {
  11. private Button btn_FirstEvent, btn_SecondEvent, btn_ThirdEvent;
  12. @Override
  13. protected void onCreate(Bundle savedInstanceState) {
  14. super.onCreate(savedInstanceState);
  15. setContentView(R.layout.activity_second);
  16. btn_FirstEvent = (Button) findViewById(R.id.btn_first_event);
  17. btn_SecondEvent = (Button) findViewById(R.id.btn_second_event);
  18. btn_ThirdEvent = (Button) findViewById(R.id.btn_third_event);
  19. btn_FirstEvent.setOnClickListener(new View.OnClickListener() {
  20. @Override
  21. public void onClick(View v) {
  22. // TODO Auto-generated method stub
  23. EventBus.getDefault().post(
  24. new FirstEvent("FirstEvent btn clicked"));
  25. }
  26. });
  27. btn_SecondEvent.setOnClickListener(new View.OnClickListener() {
  28. @Override
  29. public void onClick(View v) {
  30. // TODO Auto-generated method stub
  31. EventBus.getDefault().post(
  32. new SecondEvent("SecondEvent btn clicked"));
  33. }
  34. });
  35. btn_ThirdEvent.setOnClickListener(new View.OnClickListener() {
  36. @Override
  37. public void onClick(View v) {
  38. // TODO Auto-generated method stub
  39. EventBus.getDefault().post(
  40. new ThirdEvent("ThirdEvent btn clicked"));
  41. }
  42. });
  43. }
  44. }

3、接收

在MainActivity中,除了注册与注册,我们利用onEventMainThread(FirstEvent event)来接收来自FirstEvent的消息,使用onEventMainThread(SecondEvent event)接收来自SecondEvent 实例的消息,使用onEvent(ThirdEvent event) 来接收ThirdEvent
实例的消息。

[java] view
plain
 copy

  1. package com.harvic.tryeventbus2;
  2. import com.harvic.other.FirstEvent;
  3. import com.harvic.other.SecondEvent;
  4. import com.harvic.other.ThirdEvent;
  5. import de.greenrobot.event.EventBus;
  6. import android.app.Activity;
  7. import android.content.Intent;
  8. import android.os.Bundle;
  9. import android.util.Log;
  10. import android.view.Menu;
  11. import android.view.MenuItem;
  12. import android.view.View;
  13. import android.widget.Button;
  14. import android.widget.TextView;
  15. public class MainActivity extends Activity {
  16. Button btn;
  17. TextView tv;
  18. EventBus eventBus;
  19. @Override
  20. protected void onCreate(Bundle savedInstanceState) {
  21. super.onCreate(savedInstanceState);
  22. setContentView(R.layout.activity_main);
  23. EventBus.getDefault().register(this);
  24. btn = (Button) findViewById(R.id.btn_try);
  25. btn.setOnClickListener(new View.OnClickListener() {
  26. @Override
  27. public void onClick(View v) {
  28. // TODO Auto-generated method stub
  29. Intent intent = new Intent(getApplicationContext(),
  30. SecondActivity.class);
  31. startActivity(intent);
  32. }
  33. });
  34. }
  35. public void onEventMainThread(FirstEvent event) {
  36. Log.d("harvic", "onEventMainThread收到了消息:" + event.getMsg());
  37. }
  38. public void onEventMainThread(SecondEvent event) {
  39. Log.d("harvic", "onEventMainThread收到了消息:" + event.getMsg());
  40. }
  41. public void onEvent(ThirdEvent event) {
  42. Log.d("harvic", "OnEvent收到了消息:" + event.getMsg());
  43. }
  44. @Override
  45. protected void onDestroy() {
  46. // TODO Auto-generated method stub
  47. super.onDestroy();
  48. EventBus.getDefault().unregister(this);
  49. }
  50. }

到这里,代码就结束 了,从上面的代码,我们可以看到,EventBus是怎么接收消息的,是根据参数中类的实例的类型的判定的,所以当如果我们在接收时,同一个类的实例参数有两个函数来接收会怎样?答案是,这两个函数都会执行,下面实验一下:

在MainActivity中接收时,我们在接收SecondEvent时,在上面onEventMainThread基础上另加一个onEventBackgroundThread和onEventAsync,即下面的代码:

[java] view
plain
 copy

  1. //SecondEvent接收函数一
  2. public void onEventMainThread(SecondEvent event) {
  3. Log.d("harvic", "onEventMainThread收到了消息:" + event.getMsg());
  4. }
  5. //SecondEvent接收函数二
  6. public void onEventBackgroundThread(SecondEvent event){
  7. Log.d("harvic", "onEventBackground收到了消息:" + event.getMsg());
  8. }
  9. //SecondEvent接收函数三
  10. public void onEventAsync(SecondEvent event){
  11. Log.d("harvic", "onEventAsync收到了消息:" + event.getMsg());
  12. }

完整的代码在这里:

[java] view
plain
 copy

  1. package com.harvic.tryeventbus2;
  2. import com.harvic.other.FirstEvent;
  3. import com.harvic.other.SecondEvent;
  4. import com.harvic.other.ThirdEvent;
  5. import de.greenrobot.event.EventBus;
  6. import android.app.Activity;
  7. import android.content.Intent;
  8. import android.os.Bundle;
  9. import android.util.Log;
  10. import android.view.Menu;
  11. import android.view.MenuItem;
  12. import android.view.View;
  13. import android.widget.Button;
  14. import android.widget.TextView;
  15. public class MainActivity extends Activity {
  16. Button btn;
  17. TextView tv;
  18. EventBus eventBus;
  19. @Override
  20. protected void onCreate(Bundle savedInstanceState) {
  21. super.onCreate(savedInstanceState);
  22. setContentView(R.layout.activity_main);
  23. EventBus.getDefault().register(this);
  24. btn = (Button) findViewById(R.id.btn_try);
  25. btn.setOnClickListener(new View.OnClickListener() {
  26. @Override
  27. public void onClick(View v) {
  28. // TODO Auto-generated method stub
  29. Intent intent = new Intent(getApplicationContext(),
  30. SecondActivity.class);
  31. startActivity(intent);
  32. }
  33. });
  34. }
  35. public void onEventMainThread(FirstEvent event) {
  36. Log.d("harvic", "onEventMainThread收到了消息:" + event.getMsg());
  37. }
  38. //SecondEvent接收函数一
  39. public void onEventMainThread(SecondEvent event) {
  40. Log.d("harvic", "onEventMainThread收到了消息:" + event.getMsg());
  41. }
  42. //SecondEvent接收函数二
  43. public void onEventBackgroundThread(SecondEvent event){
  44. Log.d("harvic", "onEventBackground收到了消息:" + event.getMsg());
  45. }
  46. //SecondEvent接收函数三
  47. public void onEventAsync(SecondEvent event){
  48. Log.d("harvic", "onEventAsync收到了消息:" + event.getMsg());
  49. }
  50. public void onEvent(ThirdEvent event) {
  51. Log.d("harvic", "OnEvent收到了消息:" + event.getMsg());
  52. }
  53. @Override
  54. protected void onDestroy() {
  55. // TODO Auto-generated method stub
  56. super.onDestroy();
  57. EventBus.getDefault().unregister(this);
  58. }
  59. }

经过上面的分析,当发送SecondEvent实例的消息过来的时候,这三个函数会同时接收到并各自执行,所以当点击Second Event这个button的时候,会出现下面的结果:

好啦,这篇就到了,讲来讲去就是说一个问题:消息的接收是根据参数中的类名来决定执行哪一个的;

时间: 2024-10-26 10:41:12

EventBus使用详解(二)的相关文章

EventBus使用详解(二)——EventBus使用进阶

一.概述 前一篇给大家装简单演示了EventBus的onEventMainThread()函数的接收,其实EventBus还有另外有个不同的函数,他们分别是: 1.onEvent2.onEventMainThread3.onEventBackgroundThread4.onEventAsync 这四种订阅函数都是使用onEvent开头的,它们的功能稍有不同,在介绍不同之前先介绍两个概念:告知观察者事件发生时通过EventBus.post函数实现,这个过程叫做事件的发布,观察者被告知事件发生叫做事

UINavigationController详解二(转)页面切换和SegmentedController

原文出自:http://blog.csdn.net/totogo2010/article/details/7682433,非常感谢. 1.RootView 跳到SecondView 首先我们需要新一个View.新建SecondView,按住Command键然后按N,弹出新建页面,我们新建SecondView 2.为Button 添加点击事件,实现跳转 在RootViewController.xib中和RootViewController.h文件建立连接 在RootViewController.m

Android 布局学习之——Layout(布局)详解二(常见布局和布局参数)

[Android布局学习系列]   1.Android 布局学习之——Layout(布局)详解一   2.Android 布局学习之——Layout(布局)详解二(常见布局和布局参数)   3.Android 布局学习之——LinearLayout的layout_weight属性   4.Android 布局学习之——LinearLayout属性baselineAligned的作用及baseline    Layout Parameters(布局参数): 在XML文件中,我们经常看到类似与lay

CSS3中的弹性流体盒模型技术详解(二)

在上一篇文章<CSS3中的弹性流体盒模型技术详解(一)>里,我给大家列出了,从css1到css3各版本中盒子模型的基本元素.本篇我会把余下的属性进行详细讲解. box-pack 作用:用来规定子元素在盒子内的水平空间分配方式 box-pack 语法:box-pack: start | end | center | justify; start 对于正常方向的框,首个子元素的左边缘吸附在盒子的左边框显示 对于相反方向的框,最后子元素的右边缘吸附在盒子的右边框显示 end 对于正常方向的框,最后子

php学习之道:WSDL详解(二)

3.定义服务使用的逻辑消息 当服务的操作被调用时,服务被定义为消息交换.在wsdl文档中,这些消息被定义message元素.这些消息由称之为part元素的部分组成. 一个服务的操作,通过指定逻辑消息的方式来定义.当操作被调用时,逻辑消息被交换.(也就是说,逻辑消息代表了服务的操作)这些逻辑消息,将在网络上传输的数据定义为xml文档.他包含了所有的参数,这些参数是方法调用的一部分.(也就是说,逻辑消息里的参数,是操作对应方法的参数集合) 消息和参数列表:每一个被服务暴露的操作能且仅能有一个输入消息

LinearLayout详解二:从其父类View说起

这个View类说来就话长了,但我们又不得不说,要说呢,就得说的彻底,要让大家看得一清二楚,明明白白.所以我们就从源代码角度来看一个view是如何被加载的吧. 如果大家不知道怎么下载android的源代码,或者说懒得去下载(因为源代码确实比较大,大概有10G)的话,教大家几个取巧的办法: 1.直接在google中输入"android view.java"即可.这种方法成功率非常高,一般android的比较重要的类都能搜到. 2.给大家提供一个人家用于放源码的的git:[email pro

jquery validate 详解二

原文:http://blog.sina.com.cn/s/blog_608475eb0100h3h2.html 这里只是第二篇,前面的内容请参阅上一篇 五.常用方法及注意问题 1.用其他方式替代默认的SUBMIT 1 $().ready(function() { 2 $("#signupForm").validate({ 3 submitHandler:function(form){ 4 alert("submitted"); 5 form.submit(); 6

cocos2dx 启动过程详解二:内存管理和回调

在上一篇的第二部分中,我们有一句代码待解释的: // Draw the Scene void CCDirector::drawScene(void) { -- //tick before glClear: issue #533 if (! m_bPaused) //暂停 { m_pScheduler->update(m_fDeltaTime);   //待会会解释这里的内容 } -- } 这里是一个update函数,经常会写像this->schedule(schedule_selector(X

PopUpWindow使用详解(二)——进阶及答疑

相关文章:1.<PopUpWindow使用详解(一)——基本使用>2.<PopUpWindow使用详解(二)——进阶及答疑> 上篇为大家基本讲述了有关PopupWindow的基本使用,但还有几个相关函数还没有讲述,我们这篇将着重看看这几个函数的用法并结合源码来讲讲具体原因,最后是有关PopupWindow在使用时的疑问,给大家讲解一下. 一.常用函数讲解 这段将会给大家讲下下面几个函数的意义及用法,使用上篇那个带背景的例子为基础. [java] view plain copy pu