EventBus的简单使用与原理

一、概述

EventBus是一款针对Android优化的发布/订阅事件总线。主要功能是替代Intent,Handler,BroadCast在Fragment,Activity,Service,线程之间传递消息.优点是开销小,代码更优雅。以及将发送者和接收者解耦。

部分源码:

//通过反射,获取到订阅者的所有方法
			Method[] methods = clazz.getMethods();
			for (Method method : methods) {
				String methodName = method.getName();
				//只找以onEvent开头的方法
				if (methodName.startsWith(eventMethodName)) {
					int modifiers = method.getModifiers();
					//判断订阅者是否是public的,并且是否有修饰符,看来订阅者只能是public的,并且不能被final,static等修饰
					if ((modifiers & Modifier.PUBLIC) != 0 && (modifiers & MODIFIERS_IGNORE) == 0) {
						//获得订阅函数的参数
						Class<?>[] parameterTypes = method.getParameterTypes();
						//看了参数的个数只能是1个
						if (parameterTypes.length == 1) {
							//获取onEvent后面的部分
							String modifierString = methodName.substring(eventMethodName.length());
							ThreadMode threadMode;
							if (modifierString.length() == 0) {
								//订阅函数为onEvnet
								//记录线程模型为PostThread,意义就是发布事件和接收事件在同一个线程执行,详细可以参考我对于四个订阅函数不同点分析
								threadMode = ThreadMode.PostThread;
							} else if (modifierString.equals("MainThread")) {
								//对应onEventMainThread
								threadMode = ThreadMode.MainThread;
							} else if (modifierString.equals("BackgroundThread")) {
								//对应onEventBackgrondThread
								threadMode = ThreadMode.BackgroundThread;
							} else if (modifierString.equals("Async")) {
								//对应onEventAsync
								threadMode = ThreadMode.Async;
							} else {
								if (skipMethodVerificationForClasses.containsKey(clazz)) {
									continue;
								} else {
									throw new EventBusException("Illegal onEvent method, check for typos: " + method);
								}
							}
							//获取参数类型,其实就是接收事件的类型
							Class<?> eventType = parameterTypes[0];
							methodKeyBuilder.setLength(0);
							methodKeyBuilder.append(methodName);
							methodKeyBuilder.append('>').append(eventType.getName());
							String methodKey = methodKeyBuilder.toString();
							if (eventTypesFound.add(methodKey)) {
								// Only add if not already found in a sub class
								//封装一个订阅方法对象,这个对象包含Method对象,threadMode对象,eventType对象
								subscriberMethods.add(new SubscriberMethod(method, threadMode, eventType));
							}
						}
					} else if (!skipMethodVerificationForClasses.containsKey(clazz)) {
						Log.d(EventBus.TAG, "Skipping method (not public, static or abstract): " + clazz + "."
								+ methodName);
					}
				}
			}
			//看了还会遍历父类的订阅函数
			clazz = clazz.getSuperclass();
		}
		//最后加入缓存,第二次使用直接从缓存拿
		if (subscriberMethods.isEmpty()) {
			throw new EventBusException("Subscriber " + subscriberClass + " has no public methods called "
					+ eventMethodName);
		} else {
			synchronized (methodCache) {
				methodCache.put(key, subscriberMethods);
			}
			return subscriberMethods;
		}
	}

更多源码见:http://www.tuicool.com/articles/jUvyUjB

 


1、下载EventBus的类库
源码:https://github.com/greenrobot/EventBus

2、基本使用

(1)自定义一个类,可以是空类,比如:

 

  1. public class AnyEventType {  
  2.      public AnyEventType(){}  
  3.  }  

 

(2)在要接收消息的页面注册:

 

 
  1. eventBus.register(this);  

 

(3)发送消息

 

 
  1. eventBus.post(new AnyEventType event);  

(4)接受消息的页面实现(共有四个函数,各功能不同,这是其中之一,可以选择性的实现,这里先实现一个):

 

 
  1. public void onEvent(AnyEventType event) {}  

(5)解除注册

 
  1. eventBus.unregister(this);  

顺序就是这么个顺序,可真正让自己写,估计还是云里雾里的,下面举个例子来说明下。

 

首先,在EventBus中,获取实例的方法一般是采用EventBus.getInstance()来获取默认的EventBus实例,当然你也可以new一个又一个,个人感觉还是用默认的比较好,以防出错。

 

二、实战

先给大家看个例子:

 

当击btn_try按钮的时候,跳到第二个Activity,当点击第二个activity上面的First Event按钮的时候向第一个Activity发送消息,当第一个Activity收到消息后,一方面将消息Toast显示,一方面放入textView中显示。

按照下面的步骤,下面来建这个工程:

1、基本框架搭建

想必大家从一个Activity跳转到第二个Activity的程序应该都会写,这里先稍稍把两个Activity跳转的代码建起来。后面再添加EventBus相关的玩意。

MainActivity布局(activity_main.xml)

 

 
  1. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"  
  2.     xmlns:tools="http://schemas.android.com/tools"  
  3.     android:layout_width="match_parent"  
  4.     android:layout_height="match_parent"  
  5.     android:orientation="vertical">  
  6.       
  7.     <Button   
  8.         android:id="@+id/btn_try"  
  9.         android:layout_width="match_parent"  
  10.         android:layout_height="wrap_content"  
  11.         android:text="btn_bty"/>  
  12.     <TextView   
  13.         android:id="@+id/tv"  
  14.         android:layout_width="wrap_content"  
  15.         android:layout_height="match_parent"/>  
  16.   
  17. </LinearLayout>  

新建一个Activity,SecondActivity布局(activity_second.xml)

 
  1. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"  
  2.     xmlns:tools="http://schemas.android.com/tools"  
  3.     android:layout_width="match_parent"  
  4.     android:layout_height="match_parent"  
  5.     android:orientation="vertical"  
  6.     tools:context="com.harvic.try_eventbus_1.SecondActivity" >  
  7.   
  8.     <Button   
  9.         android:id="@+id/btn_first_event"  
  10.         android:layout_width="match_parent"  
  11.         android:layout_height="wrap_content"  
  12.         android:text="First Event"/>  
  13.   
  14. </LinearLayout>  

MainActivity.java (点击btn跳转到第二个Activity)

 
  1. public class MainActivity extends Activity {  
  2.   
  3.     Button btn;  
  4.   
  5.     @Override  
  6.     protected void onCreate(Bundle savedInstanceState) {  
  7.         super.onCreate(savedInstanceState);  
  8.         setContentView(R.layout.activity_main);  
  9.   
  10.         btn = (Button) findViewById(R.id.btn_try);  
  11.   
  12.         btn.setOnClickListener(new View.OnClickListener() {  
  13.   
  14.             @Override  
  15.             public void onClick(View v) {  
  16.                 // TODO Auto-generated method stub  
  17.                 Intent intent = new Intent(getApplicationContext(),  
  18.                         SecondActivity.class);  
  19.                 startActivity(intent);  
  20.             }  
  21.         });  
  22.     }  
  23.   
  24. }  

到这,基本框架就搭完了,下面开始按步骤使用EventBus了。

2、新建一个类FirstEvent

 

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

这个类很简单,构造时传进去一个字符串,然后可以通过getMsg()获取出来。

 

3、在要接收消息的页面注册EventBus:

在上面的GIF图片的演示中,大家也可以看到,我们是要在MainActivity中接收发过来的消息的,所以我们在MainActivity中注册消息。

通过我们会在OnCreate()函数中注册EventBus,在OnDestroy()函数中反注册。所以整体的注册与反注册的代码如下:

 

 
  1. package com.example.tryeventbus_simple;  
  2.   
  3. import com.harvic.other.FirstEvent;  
  4.   
  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.View;  
  11. import android.widget.Button;  
  12. import android.widget.TextView;  
  13. import android.widget.Toast;  
  14.   
  15. public class MainActivity extends Activity {  
  16.   
  17.     Button btn;  
  18.     TextView tv;  
  19.   
  20.     @Override  
  21.     protected void onCreate(Bundle savedInstanceState) {  
  22.         super.onCreate(savedInstanceState);  
  23.         setContentView(R.layout.activity_main);  
  24.                 //注册EventBus  
  25.         EventBus.getDefault().register(this);  
  26.   
  27.         btn = (Button) findViewById(R.id.btn_try);  
  28.         tv = (TextView)findViewById(R.id.tv);  
  29.   
  30.         btn.setOnClickListener(new View.OnClickListener() {  
  31.   
  32.             @Override  
  33.             public void onClick(View v) {  
  34.                 // TODO Auto-generated method stub  
  35.                 Intent intent = new Intent(getApplicationContext(),  
  36.                         SecondActivity.class);  
  37.                 startActivity(intent);  
  38.             }  
  39.         });  
  40.     }  
  41.     @Override  
  42.     protected void onDestroy(){  
  43.         super.onDestroy();  
  44.         EventBus.getDefault().unregister(this);//反注册EventBus  
  45.     }  
  46. }  

4、发送消息

发送消息是使用EventBus中的Post方法来实现发送的,发送过去的是我们新建的类的实例!

 

[java] view plain copy
 
 在CODE上查看代码片派生到我的代码片
  1. EventBus.getDefault().post(new FirstEvent("FirstEvent btn clicked"));  

完整的SecondActivity.Java的代码如下:

 

 
  1. package com.example.tryeventbus_simple;  
  2.   
  3. import com.harvic.other.FirstEvent;  
  4.   
  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.   
  11. public class SecondActivity extends Activity {  
  12.     private Button btn_FirstEvent;  
  13.   
  14.     @Override  
  15.     protected void onCreate(Bundle savedInstanceState) {  
  16.         super.onCreate(savedInstanceState);  
  17.         setContentView(R.layout.activity_second);  
  18.         btn_FirstEvent = (Button) findViewById(R.id.btn_first_event);  
  19.   
  20.         btn_FirstEvent.setOnClickListener(new View.OnClickListener() {  
  21.   
  22.             @Override  
  23.             public void onClick(View v) {  
  24.                 // TODO Auto-generated method stub  
  25.                 EventBus.getDefault().post(  
  26.                         new FirstEvent("FirstEvent btn clicked"));  
  27.             }  
  28.         });  
  29.     }  
  30. }  

5、接收消息

接收消息时,我们使用EventBus中最常用的onEventMainThread()函数来接收消息,具体为什么用这个,我们下篇再讲,这里先给大家一个初步认识,要先能把EventBus用起来先。

 

在MainActivity中重写onEventMainThread(FirstEvent event),参数就是我们自己定义的类:

在收到Event实例后,我们将其中携带的消息取出,一方面Toast出去,一方面传到TextView中;

[java] view plain copy
 
 在CODE上查看代码片派生到我的代码片
  1. public void onEventMainThread(FirstEvent event) {  
  2.   
  3.     String msg = "onEventMainThread收到了消息:" + event.getMsg();  
  4.     Log.d("harvic", msg);  
  5.     tv.setText(msg);  
  6.     Toast.makeText(this, msg, Toast.LENGTH_LONG).show();  
  7. }  

完整的MainActiviy代码如下:

 

 
  1. package com.example.tryeventbus_simple;  
  2.   
  3. import com.harvic.other.FirstEvent;  
  4.   
  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.View;  
  11. import android.widget.Button;  
  12. import android.widget.TextView;  
  13. import android.widget.Toast;  
  14.   
  15. public class MainActivity extends Activity {  
  16.   
  17.     Button btn;  
  18.     TextView tv;  
  19.   
  20.     @Override  
  21.     protected void onCreate(Bundle savedInstanceState) {  
  22.         super.onCreate(savedInstanceState);  
  23.         setContentView(R.layout.activity_main);  
  24.   
  25.         EventBus.getDefault().register(this);  
  26.   
  27.         btn = (Button) findViewById(R.id.btn_try);  
  28.         tv = (TextView)findViewById(R.id.tv);  
  29.   
  30.         btn.setOnClickListener(new View.OnClickListener() {  
  31.   
  32.             @Override  
  33.             public void onClick(View v) {  
  34.                 // TODO Auto-generated method stub  
  35.                 Intent intent = new Intent(getApplicationContext(),  
  36.                         SecondActivity.class);  
  37.                 startActivity(intent);  
  38.             }  
  39.         });  
  40.     }  
  41.   
  42.     public void onEventMainThread(FirstEvent event) {  
  43.   
  44.         String msg = "onEventMainThread收到了消息:" + event.getMsg();  
  45.         Log.d("harvic", msg);  
  46.         tv.setText(msg);  
  47.         Toast.makeText(this, msg, Toast.LENGTH_LONG).show();  
  48.     }  
  49.   
  50.     @Override  
  51.     protected void onDestroy(){  
  52.         super.onDestroy();  
  53.         EventBus.getDefault().unregister(this);  
  54.     }  
  55. }  

好了,到这,基本上算初步把EventBus用起来了,下篇再讲讲EventBus的几个函数,及各个函数间是如何识别当前如何调用哪个函数的。

 

如果我的文章有帮到你,请关注哦。

源码地址:http://download.csdn.net/detail/harvic880925/8111357

请大家尊重原创者版权,转载请标明出处:http://blog.csdn.net/harvic880925/article/details/40660137   谢谢!

 

posted on 2016-10-23 17:20  张诗博  阅读(1690)  评论(0编辑  收藏  举报