zoukankan      html  css  js  c++  java
  • 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   谢谢!

     

  • 相关阅读:
    求1+2+…+n, 要求不能使用乘除法、for、while、if、else、switch、case等关键字以及条件判断语句(A?B:C)。
    3,具体例子
    二十八.享元模式
    三十一.设计模式总结创建型模式
    1,学习LinQ
    软件公司需要具备什么能力的大学毕业生?
    二十九.解释器模式
    三十三.设计模式总结行为型模式
    三十二.设计模式总结结构型模式
    三十.访问者模式
  • 原文地址:https://www.cnblogs.com/zhangshibo/p/5990407.html
Copyright © 2011-2022 走看看