【攻克Android (35)】EventBus事件总线分发库

本文围绕以下六个部分展开:

一、事件总线管理
二、EventBus
三、EventBus与BroadcastReceiver的区别
案例一
案例二:一处点击发送数据,另一处或多处注册点可以及时获取更新传输过来的数据
案例三:Activity和Service之间互相发布与接收事件






一、事件总线管理

        将事件放入队列里,用于管理和分发。

        (1)保证应用的各个部分之间高效的通信及数据、事件分发。

        (2)模块间解耦:通过事件的分发,可以让各个模块间关联程序变小。

        当在开发一些庞大的的项目时,模块比较多,这个时候为了避免耦合度和保证 APP 的效率,会用到 EventBus 等类似的事件总线处理模型。这样可以简化一些数据传输操作,保证APP的简洁,做到高内聚、低耦合。


二、EventBus

        1. 概念

        EventBus是一个发布/订阅的事件总线。它可以让两个组件相互通信,但是它们之间并不相互知晓。

        EventBus模式,也被称为 Message Bus模式,或者 发布者/订阅者(publisher/subscriber)模式。

        事件响应有更多的线程选择,EventBus 可以向不同的线程中发布事件。

        EventBus 支持 Sticky Event。



        2. 有3个主要的元素:

        (1)Event:事件。

        Event可以是任意类型的对象。

        (2)Subscriber:事件订阅者,接收特定的事件。

        在EventBus中,使用约定来指定事件订阅者以简化使用。即,所有事件订阅都是以onEvent开头的函数。具体来说,函数的名字是onEventMainThread,onEventPostThread,onEventBackgroundThread,onEventAsync这四个,onEvent是默认的接收数据处理的方法。这四个事件订阅函数,每个都与一个“ThreadMode”相关联,ThreadMode指定了会调用的函数。有以下4个ThreadMode:

        1)PostThread:事件的处理和事件的发送在相同的进程。

        事件处理时间不应太长,不然影响事件的发送线程,而这个线程可能是UI线程。对应的函数名是onEvent。

        2)MainThread:事件的处理会在UI线程中执行。

        事件处理时间不应太长,长了会ANR的。对应的函数名是onEventMainThread。

        3)BackgroundThread:事件的处理会在一个后台线程中执行。

        虽然事件处理是在后台线程,但事件处理时间还是不应该太长,因为如果发送事件的线程是后台线程,会直接执行事件。如果当前线程是UI线程,事件会被加到一个队列中,由一个线程依次处理这些事件,如果某个事件处理时间太长,会阻塞后面的事件的派发或处理。对应的函数名是onEventBackgroundThread。

        4)AsyncThread:事件处理会在单独的线程中执行。

        主要用于在后台线程中执行耗时操作,每个事件会开启一个线程(有线程池),但最好限制线程的数目。

        根据事件订阅的函数名称的不同,会使用不同的ThreadMode。比如在后台线程加载了数据想在UI线程显示,订阅者只需把函数命名为onEventMainThread。

        3. 基本用法

        注册订阅发布注销注册 等步骤。使用时需先注册、订阅,然后向订阅者发布消息数据即可。

        (1)注册(3个构造方法):

        EventBus.getDefault().register(this);

        EventBus.getDefault().register(new MyClass());

        // 注册三个参数分别是:消息订阅者(接收者),接收方法名,事件类。         EventBus.getDefault().register(this,"setTextA",SetTextAEvent.class);

        (2)注销注册:

        EventBus.getDefault().unregister(this);

        EventBus.getDefault().unregister(new MyClass());

        (3)定义事件类型:

        public class MyEvent {}

        (4)订阅处理数据:

        public void onEvent(AnyEventType event){}:默认的接收数据处理的方法

        public void onEventMainThread{}

        onEventPostThread

        onEventBackgroundThread

        onEventAsync

        (5)发布:

        EventBus.getDefault().postSticky(new SecondActivityEvent("Message from SecondActivity"));

        EventBus.getDefault().post(new ChangeImgEvent(1));


三、EventBus与BroadcastReceiver的区别

        1. 二者类似。

        在Android中广播分为两个方面:广播发送者和广播接收者,通常情况下,BroadcastReceiver指的就是广播接收者(广播接收器)。

        广播:在一个地方注册广播,在另一个地方针对action发送广播、传递数据,对应注册的地方就可以收到广播。

        EventBus:基于订阅/发布的模式。在需要接收数据的地方,进行注册与订阅,在需要发布数据的地方发布,则在注册的地方就可以收到了。

        简单点说,就是两人约定好怎么通信,一人发布消息,另外一个约定好的人立马接收到你发的消息。EventBus就可以帮助减少做很多事,不管你在任何地方任何位置发布一个事件,接收者都能立马接收到你的消息,不用你考虑android子线程操作UI线程的问题。

        2. 二者区别。

        (1)EventBus 有三个主要的元素:事件、订阅和发布。广播两个元素:订阅和发布,但是广播是针对整个App而言的。

        (2)BroadcastReceiver是组件,需要在功能清单中注册。而EventBus 不需要注册。

        (3)BroadcastReceiver只能做一件事情,而EventBus多事件处理比较好。

        (4)在不同场景中的适用性:

            1)同一App内部的同一组件内的消息通信(单个或多个线程之间),实际应用中肯定是不会用到广播机制的(虽然可以用),无论是使用扩展变量作用域、基于接口的回调还是Handler-post/Handler-Message等方式,都可以直接处理此类问题,若使用广播机制,显然有些“杀鸡牛刀”的感觉。

            2)同一app内部的不同组件之间的消息通信(单个进程),对于此类需求,在有些教复杂的情况下单纯的依靠基于接口的回调等方式不好处理,此时可以直接使用EventBus等,相对而言,EventBus由于是针对同一进程,用于处理此类需求非常适合,且轻松。

            3)其他情形,由于涉及不同进程间的消息通信,此时根据实际业务使用广播机制会显得非常适宜。


案例一

        MainActivity

public class MainActivity extends Activity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        // 1. 注册
        EventBus.getDefault().register(this);
    }

    /**
     * 3. 发送数据消息事件
     */
    private void postData() {
        String string = "我是消息";
        EventBus.getDefault().post(string);
    }

    /**
     * 2. 接收数据消息事件
     */
    public void onEvent(String string) {

    }

    public void onEventMainThread(String string) {

    }

    public void onEventPostThread(String string) {

    }

    public void onEventBackgroundThread(String string) {

    }

    public void onEventAsync(String string) {

    }


    @Override
    protected void onDestroy() {
        super.onDestroy();
        // 4. 取消注册
        EventBus.getDefault().unregister(this);
    }
    
}



案例二:一处点击发送数据,另一处或多处注册点可以及时获取更新传输过来的数据


        1. activity_main.xml。写一个按钮和一个文本框。

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:paddingBottom="@dimen/activity_vertical_margin"
    android:paddingLeft="@dimen/activity_horizontal_margin"
    android:paddingRight="@dimen/activity_horizontal_margin"
    android:paddingTop="@dimen/activity_vertical_margin"
    tools:context=".MainActivity">

    <Button
        android:id="@+id/btnSend"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="发送事件"/>

    <TextView
        android:id="@+id/tvContent"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_below="@id/btnSend"
        android:gravity="center"
        android:textSize="20sp"/>

</RelativeLayout>


        2. MainActivity。声明和初始化两个控件。

    private TextView tvContent;
    private Button btnSend;


        tvContent = (TextView) findViewById(R.id.tvContent);
        btnSend = (Button) findViewById(R.id.btnSend);



        3. MainActivity。onCreate()方法中注册。

EventBus.getDefault().register(this);


        4. MainActivity。onDestroy()方法中取消注册。

    @Override
    protected void onDestroy() {
        super.onDestroy();
        EventBus.getDefault().unregister(this);
    }


        5. MainActivity。通过事件监听器设置按钮的点击事件。

        btnSend.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {

            }
        });


        6. 创建自定义的事件类:MyEvent。

package com.android.eventbustwo;

/**
 * 自定义的事件类
 */
public class MyEvent {
    private String type;
    private String content;

    public String getType() {
        return type;
    }

    public void setType(String type) {
        this.type = type;
    }

    public String getContent() {
        return content;
    }

    public void setContent(String content) {
        this.content = content;
    }
}


        7. MainActivity。在按钮点击事件中,创建自定义事件类MyEvent的实例事件,并当点击按钮的时候,发送事件。

        btnSend.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                MyEvent event = new MyEvent();
                event.setType("0");
                event.setContent("0内容");
                // 发送数据事件
                EventBus.getDefault().post(event);
            }
        });




        8. MainActivity。通过onEvent()订阅(接收)事件。

    public void onEvent(MyEvent event){
        if(event.getType().equals("0")){
            tvContent.setText(event.getContent());
        }
    }


        运行效果如下:



        9. MainActivity。在按钮点击事件中,将事件类型和内容改变,并当点击按钮的时候,发送事件。

        btnSend.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                MyEvent event = new MyEvent();
                event.setType("1");
                event.setContent("1内容");
                // 发送数据事件
                EventBus.getDefault().post(event);
            }
        });


        10. MainActivity。通过onEventMainThread()同样可以订阅(接收)事件。

    public void onEventMainThread(MyEvent event){
        if(event.getType().equals("1")){
            tvContent.setText(event.getContent());
        }
    }


        运行效果如下:




        代码补充:

        1. activity_main.xml

        2. MainActivity

package com.android.eventbustwo;

import android.app.Activity;
import android.os.Bundle;
import android.view.Menu;
import android.view.MenuItem;
import android.view.View;
import android.widget.Button;
import android.widget.TextView;

import de.greenrobot.event.EventBus;

public class MainActivity extends Activity {
    // 2.
    private TextView tvContent;
    private Button btnSend;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        // 3.
        tvContent = (TextView) findViewById(R.id.tvContent);
        btnSend = (Button) findViewById(R.id.btnSend);
        // 6.
        btnSend.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                // 8.
                MyEvent event = new MyEvent();
                event.setType("0");
                event.setContent("0内容");
                // 发送数据事件
                EventBus.getDefault().post(event);
            }
        });
        // 4.
        EventBus.getDefault().register(this);
    }

    /**
     * 9.
     * @param event
     */
    public void onEvent(MyEvent event){
        if(event.getType().equals("0")){
            tvContent.setText(event.getContent());
        }
    }

    /**
     * 10.
     * @param event
     */
    public void onEventMainThread(MyEvent event){
        if(event.getType().equals("1")){
            tvContent.setText(event.getContent());
        }
    }

    /**
     * 5.
     */
    @Override
    protected void onDestroy() {
        super.onDestroy();
        EventBus.getDefault().unregister(this);
    }

    // -----------------------------------------------------------
    @Override
    public boolean onCreateOptionsMenu(Menu menu) {
        // Inflate the menu; this adds items to the action bar if it is present.
        getMenuInflater().inflate(R.menu.menu_main, menu);
        return true;
    }

    @Override
    public boolean onOptionsItemSelected(MenuItem item) {
        // Handle action bar item clicks here. The action bar will
        // automatically handle clicks on the Home/Up button, so long
        // as you specify a parent activity in AndroidManifest.xml.
        int id = item.getItemId();

        //noinspection SimplifiableIfStatement
        if (id == R.id.action_settings) {
            return true;
        }

        return super.onOptionsItemSelected(item);
    }
}


        3. MyEvent

package com.android.eventbustwo;

/**
 * 7. 自定义的事件类
 */
public class MyEvent {
    private String type;
    private String content;

    public String getType() {
        return type;
    }

    public void setType(String type) {
        this.type = type;
    }

    public String getContent() {
        return content;
    }

    public void setContent(String content) {
        this.content = content;
    }
}



案例三:Activity和Service之间互相发布与接收事件

        主活动有4家公司名显示,当在文本框中输入“Amazon”,点击按钮后,将新的公司名显示在主活动界面上。







        1. 创建一个Service: EventService。

        2. EventService。将onBind中的异常去掉,改为 返回null。

    @Override
    public IBinder onBind(Intent intent) {
        return null;
    }


        3. activity_main.xml。一个显示的文本框,用于显示Service发布的事件;一个可编辑文本框,用于输入字符串,并发布事件到Service;一个按钮,通过点击按钮,发布事件,然后Service进行订阅。



<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:paddingBottom="@dimen/activity_vertical_margin"
    android:paddingLeft="@dimen/activity_horizontal_margin"
    android:paddingRight="@dimen/activity_horizontal_margin"
    android:paddingTop="@dimen/activity_vertical_margin"
    tools:context=".MainActivity">

    <TextView
        android:id="@+id/tvCompany"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="@string/hello_world" />

    <EditText
        android:id="@+id/txtCompany"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_below="@id/tvCompany"
        android:hint="@string/txt_company" />

    <Button
        android:id="@+id/btnCompany"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_below="@id/txtCompany"
        android:text="@string/btn_send_company" />

</RelativeLayout>


        4. EventService。重写onCreate()、onStartCommand()、onDestroy()方法。

        5. EventService。声明一个 List 集合,用于放公司名的集合,并在onCreate()方法中创建实例(ArrayList类型)。

    private List<String> list;


        list = new ArrayList<>();


        6. EventService。在onStartCommand()方法中,往List集合中添加元素(公司名)。销毁(onDestroy())的时候,集合设为null。

        list.add("Google");
        list.add("Facebook");
        list.add("Twitter");
        list.add("Apple");


        list = null;


        8. EventService。onCreate()中注册EventBus,onDestroy()销毁时注销注册。

        EventBus.getDefault().register(this);


        EventBus.getDefault().unregister(this);


        9. MainActivity。onCreate()中同样注册EventBus,onDestroy()销毁时同样注销注册。

        10. 新建一个目录:event,用来放事件类。

        11. 在event目录下,新建一个用于从 Service 发布集合数据(公司名的集合)集合事件类:ListEvent。

public class ListEvent {
    // 集合
    private List<String> list;

    // 含参构造方法
    public ListEvent(List<String> list) {
        this.list = list;
    }

    // getter方法(只需getter方法)
    public List<String> getList() {
        return list;
    }
}


        12. 在event目录下,新建一个用于从 Activity 发布字符串(公司名)的集合事件类:CompanyEvent。

public class CompanyEvent {
    // 字符串 company
    private String company;

    // 含参构造方法
    public CompanyEvent(String company) {
        this.company = company;
    }

    // getter方法(只需getter方法)
    public String getCompany() {
        return company;
    }
}


        13. MainActivity。在按钮点击事件中,发布事件。

    @OnClick(R.id.btnCompany)
    public void btnCompanyClick() {
        // 获得 在 txtCompany 文本框中输入的 company 字符串 的值
        String company = txtCompany.getText().toString();
        // Activity 发布事件 --- 将 在 txtCompany 文本框中输入的 company 字符串,
        //                          使用 CompanyEvent 发布过去
        EventBus.getDefault().post(new CompanyEvent(company));
        // 发布之后,将 文本框内容清空
        txtCompany.setText("");
    }


        14. EventService。Service订阅事件,将Activity发布的公司名加入List集合中。然后,再将最新的公司名List集合发布出去。

    /**
     * Service 订阅事件
     * @param event
     */
    public void onEvent(CompanyEvent event){
        // 将 事件中传过来的、在 txtCompany 文本框中输入的 company 字符串,放入 list集合中
        list.add(event.getCompany());
        // 再将 list集合,发布给 Activity,
        // 这样,Activity订阅后,就可以在 控制台和界面上 显示最新的 list集合的值
        EventBus.getDefault().postSticky(new ListEvent(list));
    }


        15. MainActivity。Activity 订阅了ListEvent事件,所以接收到了Service发布的公司名List集合的值,然后将其显示在控制台和界面的文本上。

    /**
     * Activity 订阅事件
     *
     * @param event
     */
    public void onEvent(ListEvent event) {
        // 从Service发布的事件中获得 参数值(list集合)
        List<String> list = event.getList();
        // 日志输出 list集合的值 (控制台显示 list集合的值)
        Log.v(TAG, list.toString());
        // 将 list集合的值 显示在 tvCompany 文本中。(界面的文本上 显示 list集合的值)
        tvCompany.setText(list.toString());
    }



        代码目录结构:



        补充代码:

        (1)EventService。

package com.android.eventbusservice;

import android.app.Service;
import android.content.Intent;
import android.os.IBinder;
import android.util.Log;

import com.android.eventbusservice.event.CompanyEvent;
import com.android.eventbusservice.event.ListEvent;

import java.util.ArrayList;
import java.util.List;

import de.greenrobot.event.EventBus;

public class EventService extends Service {
    private static final String TAG = "MainActivity";
    // 声明一个 List 集合 -- 用于放公司名的集合
    private List<String> list;

    public EventService() {
    }


    @Override
    public void onCreate() {
        super.onCreate();
        Log.v(TAG,"Service.onCreate");
        // 当前组件注册事件
        // (注册事件后,一定要有 onEvent() 方法,否则报错)
        EventBus.getDefault().register(this);
        // 创建实例 - ArrayList类型
        list = new ArrayList<>();
    }

    /**
     * 此方法系统自动调用,不写也可。可把里面的代码写入 onCreate()里面。
     */
    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {
        Log.v(TAG,"Service.onStartCommand");
        // 往集合中添加元素
        list.add("Google");
        list.add("Facebook");
        list.add("Twitter");
        list.add("Apple");

        // Service 发布事件 -- 将 list 集合,使用 ListEvent 发送过去
        //                (ListEvent:用于从 Service 传事件到 Activity)
        EventBus.getDefault().postSticky(new ListEvent(list));
        // 改为:返回 START_STICKY
        return START_STICKY;
    }

    @Override
    public void onDestroy() {
        super.onDestroy();
        Log.v(TAG, "Service.onDestroy");
        // 销毁的时候,取消事件
        EventBus.getDefault().unregister(this);
        // 销毁的时候,集合设为空
        list = null;
    }

    /**
     * Service 订阅事件
     * @param event
     */
    public void onEvent(CompanyEvent event){
        // 将 事件中传过来的、在 txtCompany 文本框中输入的 company 字符串,放入 list集合中
        list.add(event.getCompany());
        // 再将 list集合,发布给 Activity,
        // 这样,Activity订阅后,就可以在 控制台和界面上 显示最新的 list集合的值
        EventBus.getDefault().postSticky(new ListEvent(list));
    }

    @Override
    public IBinder onBind(Intent intent) {
        return null;
    }
}


        (2)MainActivity。

package com.android.eventbusservice;

import android.app.Activity;
import android.content.Intent;
import android.os.Bundle;
import android.util.Log;
import android.view.Menu;
import android.view.MenuItem;
import android.widget.EditText;
import android.widget.TextView;

import com.android.eventbusservice.event.CompanyEvent;
import com.android.eventbusservice.event.ListEvent;

import java.util.List;

import butterknife.ButterKnife;
import butterknife.InjectView;
import butterknife.OnClick;
import de.greenrobot.event.EventBus;

public class MainActivity extends Activity {
    private static final String TAG = "MainActivity";

    @InjectView(R.id.tvCompany)
    TextView tvCompany;
    @InjectView(R.id.txtCompany)
    EditText txtCompany;


    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        ButterKnife.inject(this);

        // 在 Activity 中同样注册事件
        EventBus.getDefault().register(this);

        // 启动服务 -- 使用 startService 方式启动
        // 启动服务后,activity与service脱钩了。
        // 但是可以通过黄油刀进行联系。
        startService(new Intent(this, EventService.class));
    }
    
    @Override
    protected void onDestroy() {
        super.onDestroy();
        // 在 Activity 中同样注销事件
        EventBus.getDefault().unregister(this);
    }

    @OnClick(R.id.btnCompany)
    public void btnCompanyClick() {
        // 获得 在 txtCompany 文本框中输入的 company 字符串 的值
        String company = txtCompany.getText().toString();
        // Activity 发布事件 --- 将 在 txtCompany 文本框中输入的 company 字符串,
        //                          使用 CompanyEvent 发布过去
        EventBus.getDefault().post(new CompanyEvent(company));
        // 发布之后,将 文本框内容清空
        txtCompany.setText("");
    }

    /**
     * Activity 订阅事件
     *
     * @param event
     */
    public void onEvent(ListEvent event) {
        // 从Service发布的事件中获得 参数值(list集合)
        List<String> list = event.getList();
        // 日志输出 list集合的值 (控制台显示 list集合的值)
        Log.v(TAG, list.toString());
        // 将 list集合的值 显示在 tvCompany 文本中。(界面的文本上 显示 list集合的值)
        tvCompany.setText(list.toString());
    }

    // ---------------------------------------------------------
    @Override
    public boolean onCreateOptionsMenu(Menu menu) {
        // Inflate the menu; this adds items to the action bar if it is present.
        getMenuInflater().inflate(R.menu.menu_main, menu);
        return true;
    }

    @Override
    public boolean onOptionsItemSelected(MenuItem item) {
        // Handle action bar item clicks here. The action bar will
        // automatically handle clicks on the Home/Up button, so long
        // as you specify a parent activity in AndroidManifest.xml.
        int id = item.getItemId();

        //noinspection SimplifiableIfStatement
        if (id == R.id.action_settings) {
            return true;
        }

        return super.onOptionsItemSelected(item);
    }
}



整理时重点参考:http://www.jikexueyuan.com/course/933.html


猜你喜欢

转载自xiangdonglee.iteye.com/blog/2235339
今日推荐