今天,简单讲讲Android里关于EventBus的使用。

这几天,由于面试的缘故,我听到了很多Android的流行框架,但是之前自己在公司做APP时并没有使用,所以没有了解。于是在网上查找了资料,学习了这些Android的流行框架的使用,准备做一个Android流行框架的专栏。分别介绍Android框架:EventBus,RXJava,Retrofit,OKHttp,Dagger2,ButterKnife,Glide,volle,green这些框架。目前只打算将这些框架的基本使用,以后再项目里用到,才可能深入了解。有兴趣的可以关注一下。

一、简介

EventBus是由greenrobot 组织贡献的一个Android事件发布/订阅轻量级框架。EventBus是一个Android端优化的publish/subscribe消息总线,简化了应用程序内各组件间、组件与后台线程间的通信。比如请求网络,等网络返回时通过Handler或Broadcast通知UI,两个Fragment之间需要通过Listener通信,这些需求都可以通过EventBus实现。

官网地址:http://greenrobot.org/eventbus/
翻译:http://blog.csdn.net/poorkick/article/details/55099311

二、添加依赖

compile 'org.greenrobot:eventbus:3.0.0'

三、解锁技能

  1. EventBus的三要素

    • Event:事件,可以是任意类型的对象。
    • Subscriber:事件订阅者,在EventBus3.0之前消息处理的方法只能限定于onEvent、onEventMainThread、onEventBackgroundThread和onEventAsync,他们分别代表四种线程模型。而在EventBus3.0之后,事件处理的方法可以随便取名,但是需要添加一个注解@Subscribe,并且要指定线程模型(默认为POSTING)。
    • Publisher:事件发布者,可以在任意线程任意位置发送事件,直接调用EventBus的post(Object)方法。可以自己实例化EventBus对象,但一般使用EventBus.getDefault()就好了,根据post函数参数的类型,会自动调用订阅相应类型事件的函数。
  2. EventBus的四种线程模型(ThreadMode)
    • POSTING(默认):如果使用事件处理函数指定了线程模型为POSTING,那么该事件在哪个线程发布出来的,事件处理函数就会在这个线程中运行,也就是说发布事件和接收事件在同一个线程。在线程模型为POSTING的事件处理函数中尽量避免执行耗时操作,因为它会阻塞事件的传递,甚至有可能会引起应用程序无响应(ANR)。
    • MAIN:事件的处理会在UI线程中执行。事件处理时间不能太长,长了会ANR的。
    • BACKGROUND:如果事件是在UI线程中发布出来的,那么该事件处理函数就会在新的线程中运行,如果事件本来就是子线程中发布出来的,那么该事件处理函数直接在发布事件的线程中执行。在此事件处理函数中禁止进行UI更新操作。
    • ASYNC:无论事件在哪个线程发布,该事件处理函数都会在新建的子线程中执行,同样,此事件处理函数中禁止进行UI更新操作。
  3. 使用步骤

    • 注册:EventBus.getDefault().register(this);
    • 解注册(为防止内存泄漏):EventBus.getDefault().unregister(this);
    • 构造发送消息类

  1. public class MessageEvent {
  2. public String name;
  3. public String password;
  4.  
  5. public MessageEvent(String name, String password) {
  6. this.name = name;
  7. this.password = password;
  8. }
  9. }

4.发布消息:EventBus.getDefault().post(new MessageEvent(“name”,”password”));
5.接收消息:可以有四种线程模型选择

  1. @Subscribe(threadMode = ThreadMode.MAIN)
  2. public void messageEventBus(MessageEvent event){
  3. tv_result.setText("name:"+event.name+" passwrod:"+event.password);
  4. }

4.粘性事件 
   之前说的使用方法,都是需要先注册(register),再post,才能接受到事件;如果你使用postSticky发送事件,那么可以不需要先注册,也能接受到事件,也就是一个延迟注册的过程。 
 
 普通的事件我们通过post发送给EventBus,发送过后之后当前已经订阅过的方法可以收到。但是如果有些事件需要所有订阅了该事件的方法都能执行呢?例如一个Activity,要求它管理的所有Fragment都能执行某一个事件,但是当前我只初始化了3个Fragment,如果这时候通过post发送了事件,那么当前的3个Fragment当然能收到。但是这个时候又初始化了2个Fragment,那么我必须重新发送事件,这两个Fragment才能执行到订阅方法。 
   粘性事件就是为了解决这个问题,通过 postSticky 发送粘性事件,这个事件不会只被消费一次就消失,而是一直存在系统中,知道被
removeStickyEvent 删除掉。那么只要订阅了该粘性事件的所有方法,只要被register
的时候,就会被检测到,并且执行。订阅的方法需要添加 sticky = true 属性。

  1. 构造发送信息类
      1. public class StickyEvent {
      2. public String msg;
      3.  
      4. public StickyEvent(String msg) {
      5. this.msg = msg;
      6. }
      7. }

2.发布消息:EventBus.getDefault().postSticky(new StickyEvent(“我是粘性事件”));
 3.接收消息:和之前的方法一样,只是多了一个 sticky = true 的属性

  1. @Subscribe(threadMode = ThreadMode.MAIN, sticky = true)
  2. public void onEvent(StickyEvent event){
  3. tv_c_result.setText(event.msg);
  4. }

4.注册:

EventBus.getDefault().register(CActivity.this);

5.解注册:

  1. EventBus.getDefault().removeAllStickyEvents();
  2. EventBus.getDefault().unregister(CActivity.class);

四、举个栗子

1.MainActivity.class

  1. package com.example.wgh.eventbusdemo;
  2.  
  3. import android.app.Activity;
  4. import android.content.Intent;
  5. import android.os.Bundle;
  6. import android.view.View;
  7. import android.widget.ProgressBar;
  8.  
  9. import org.greenrobot.eventbus.EventBus;
  10. import org.greenrobot.eventbus.Subscribe;
  11. import org.greenrobot.eventbus.ThreadMode;
  12.  
  13.  
  14. public class MainActivity extends Activity {
  15.  
  16. public ProgressBar progressBar = null;
  17. public int time = 0;
  18. @Override
  19. protected void onCreate(Bundle savedInstanceState) {
  20. super.onCreate(savedInstanceState);
  21. setContentView(R.layout.activity_main);
  22.  
  23. findViewById(R.id.button).setOnClickListener(new View.OnClickListener() {
  24. @Override
  25. public void onClick(View v) {
  26. new Thread(new Runnable() {
  27. @Override
  28. public void run() {
  29. while (time<100){
  30. time += 15;
  31. EventBus.getDefault().post(new TestEvent(time));
  32. try {
  33. Thread.sleep(200);
  34. } catch (InterruptedException e) {
  35. e.printStackTrace();
  36. }
  37. }
  38. }
  39. }).start();
  40. }
  41. });
  42. progressBar = (ProgressBar) findViewById(R.id.progressbar);
  43.  
  44. EventBus.getDefault().register(this);
  45. }
  46.  
  47. @Override
  48. protected void onDestroy() {
  49. super.onDestroy();
  50. EventBus.getDefault().unregister(this);
  51. }
  52. @Subscribe(threadMode = ThreadMode.MAIN)
  53. public void onEventMainThread(TestEvent event){
  54. progressBar.setProgress(event.getMsg());
  55. }
  56. }

2.activity_main.xml

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
  3. xmlns:tools="http://schemas.android.com/tools"
  4. android:id="@+id/activity_main"
  5. android:layout_width="match_parent"
  6. android:layout_height="match_parent"
  7. tools:context="com.example.wgh.eventbusdemo.MainActivity"
  8. android:orientation="vertical">
  9.  
  10. <ProgressBar
  11. android:id="@+id/progressbar"
  12. android:layout_width="match_parent"
  13. android:layout_height="wrap_content"
  14. android:layout_marginTop="150dp"
  15. android:max="100"
  16. style="@style/Widget.AppCompat.ProgressBar.Horizontal"/>
  17. <Button
  18. android:id="@+id/button"
  19. android:layout_marginTop="10dp"
  20. android:layout_width="match_parent"
  21. android:layout_height="wrap_content"
  22. android:text="开始下载"/>
  23. </LinearLayout>

3.TestEvent.class

  1. public class TestEvent {
  2. private int mMsg;
  3. public TestEvent(int msg) {
  4. mMsg = msg;
  5. }
  6. public int getMsg(){
  7. return mMsg;
  8. }
  9. }

简单讲讲,EventBus其实就是一个框架,用来取代广播,intent再界面之间传值的。我们只要导入类库,然后创建自己的消息类,再activity注册EventBus,然后通过EventBus.getDefault().post(new TestEvent(time))就可以发生消息,再函数@Subscribe(threadMode = ThreadMode.MAIN)//表面就收的代码在主线程运行
    public void onEventMainThread(TestEvent event){
        progressBar.setProgress(event.getMsg());
    }内接受消息就可以了。activity销毁时记得取消EventBus的注册。

android EventBus的简单使用就讲完了。

就这么简单。

android EventBus的简单使用的更多相关文章

  1. Android 第三方类库简单使用之EventBus

    Android 第三方类库之EventBus 1 PS 工欲善其事必先利其器. Eventbus也是一款在开发中常用的利器 这篇也对EventBus的简单介绍和使用,与之前个xutils介绍的级别一样 ...

  2. Android EventBus 3.0.0 使用总结

    转载请标明出处:http://www.cnblogs.com/zhaoyanjun/p/6039221.html 本文出自[赵彦军的博客] 前言 EventBus框架 EventBus是一个通用的叫法 ...

  3. Android EventBus实战 没听过你就out了

    转载请表明出处:http://blog.csdn.net/lmj623565791/article/details/40794879,本文出自:[张鸿洋的博客] 1.概述 最近大家面试说经常被问到Ev ...

  4. Android EventBus源码解析 带你深入理解EventBus

    转载请标明出处:http://blog.csdn.net/lmj623565791/article/details/40920453,本文出自:[张鸿洋的博客] 上一篇带大家初步了解了EventBus ...

  5. android EventBus 的使用

    今天简单的介绍 一下啊 android  EventBus 的使用 EventBus 在官方介绍中是订阅......什么的 一大堆  ,  在我android 菜鸟眼里 就是用来代替android 广 ...

  6. Android EventBus源代码解析 带你深入理解EventBus

    转载请标明出处:http://blog.csdn.net/lmj623565791/article/details/40920453,本文出自:[张鸿洋的博客] 上一篇带大家初步了解了EventBus ...

  7. Android EventBus现实 听说你out该

    转载请注明出处:http://blog.csdn.net/lmj623565791/article/details/40794879.本文出自:[张鸿洋的博客] 1.概述 近期大家面试说常常被问到Ev ...

  8. Android EventBus 3.0 实例使用详解

    EventBus的使用和原理在网上有很多的博客了,其中泓洋大哥和启舰写的非常非常棒,我也是跟着他们的博客学会的EventBus,因为是第一次接触并使用EventBus,所以我写的更多是如何使用,源码解 ...

  9. Android -- EventBus解析

    EventBus EventBus 是一个 Android 事件发布/订阅框架,通过解耦发布者和订阅者简化 Android 事件传递.传统的事件传递方式包括:Handler.BroadCastRece ...

随机推荐

  1. iOS之事件的传递和响应机制

    前言: 按照时间顺序,事件的生命周期是这样的: 事件的产生和传递(事件如何从父控件传递到子控件并寻找到最合适的view.寻找最合适的view的底层实现.拦截事件的处理)->找到最合适的view后 ...

  2. jQuery-PHP跨域请求数据

    jQuery: //获取域名 function getDomain(url){ var a = document.createElement('a'); a.href = url; url=a.hos ...

  3. 从“关于Java堆与栈的思考”一帖看错误信息的传播

    我对转贴的信息一直有敌意,原因如下:首先,除了制造更多的信息垃圾,转贴不会带来新的价值,想收藏的话一个链接足矣:其次,将错误信息以讹传讹,混淆视听.不妨选一个典型的例子说明一二. 相信<关于Ja ...

  4. window 注册表上下文菜单如何配置?

    注册表结构? Keys Abbreviation Description 描述 HKEY_CLASSES_ROOT HKCR Stores file association and COM objec ...

  5. MySQL错误日志提示innodb_table_stats和innodb_index_stats不存在故障处理

    查看MySQL error日志,发现有如下报错 7efbc586f700 InnoDB: Error: Table "mysql"."innodb_table_stats ...

  6. 003-ARP地址解析协议

    一.概念 地址解析协议,即ARP(Address Resolution Protocol),是根据IP地址获取物理地址的一个TCP/IP协议.主机发送信息时将包含目标IP地址的ARP请求广播到网络上的 ...

  7. JavaWeb—拦截器Interceptor

    1.概念 java里的拦截器是动态拦截Action调用的对象,它提供了一种机制可以使开发者在一个Action执行的前后执行一段代码,也可以在一个Action执行前阻止其执行,同时也提供了一种可以提取A ...

  8. JavaScript:学习笔记(2)——基本概念与数据类型

    JavaScript:学习笔记(2)——基本概念与数据类型 语法 1.区分大小写.Test 和 test 是完全不同的两个变量. 2.语句最好以分号结束,也就是说不以分号结束也可以. 变量 1.JS的 ...

  9. range基础

    collapse这个方法是把结束位置抛弃掉,并不是简单的设置到开始位置. 结束位置被抛弃掉以后,只要没有给它重新设置位置,它就一直都会等 于开始位置.即使你修改了开始位置,结束位置还是会在修改后的开始 ...

  10. iOS 根据农历日期 获取当前的农历年份 即 干支纪年法算农历年

    前言:我国古代是用干支纪年的,近代史上提到的甲午战争.戊戌变法.辛亥革命等名词就是干支纪年.所谓干支就是十天干和十二地支的简称.天干.地支按照一定规则(单配单,双配双)可以搭配成60对,也就是一个甲子 ...