EventBus与RxBus区别

先介绍一下EventBus

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

官网地址:http://greenrobot.org/eventbus/ 


三、解释技能

  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);
    • 构造发送消息类

public class MessageEvent {
    public String name;
    public String password;

    public MessageEvent(String name, String password) {
        this.name = name;
        this.password = password;
    }
}
  • 发布消息:EventBus.getDefault().post(new MessageEvent(“name”,”password”));
  • 接收消息:可以有四种线程模型选择

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

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

  • 构造发送信息类

public class StickyEvent {
    public String msg;

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

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

注册

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

解注册

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

四、举个栗子

  1. 主线程发送事件:

    • 自定义事件(类似定义JavaBean),包含用户的姓名和密码;


public class UserEvent {
    private String name;
    private String password;

    public UserEvent() {
    }

    public UserEvent(String name, String password) {
        this.name = name;
        this.password = password;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public String getPassword() {
        return password;
    }

    public void setPassword(String password) {
        this.password = password;
    }

    @Override
    public String toString() {
        return "UserEvent{" +
                "name='" + name + '\'' +
                ", password='" + password + '\'' +
                '}';
    }
}

在onCreate方法中注册订阅者,在onDestroy中解注册。

public class MainActivity extends AppCompatActivity {

    @BindView(R.id.jump)
    Button mJump;
    @BindView(R.id.send)
    Button mSend;
    @BindView(R.id.tv_result)
    TextView mTvResult;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        ButterKnife.bind(this);
        //注册订阅者
        EventBus.getDefault().register(this);
    }

    @OnClick({R.id.jump, R.id.send})
    public void onViewClicked(View view) {
        switch (view.getId()) {
            case R.id.jump:
                startActivity(new Intent(MainActivity.this, SecActivity.class));
                break;
            case R.id.send:
                break;
        }
    }

    //定义处理接收的方法
    @Subscribe(threadMode = ThreadMode.MAIN)
    public void userEventBus(UserEvent userEvent){
        mTvResult.setText(userEvent.toString());
    }

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

在另一个activity中发送事件,让订阅者能够接收;

@OnClick({R.id.sendData, R.id.receive})
public void onViewClicked(View view) {
    switch (view.getId()) {
        case R.id.sendData:
            //发送事件
            EventBus.getDefault().post(new UserEvent("Mr.sorrow", "123456"));
            finish();
            break;
        case R.id.receive:
            break;
    }
}

发送粘性事件:

  • MainActivity中发送粘性事件;

case R.id.send:
        EventBus.getDefault().postSticky(new MessageEvent("粘性事件", "urgent"));
        startActivity(new Intent(MainActivity.this, SecActivity.class));
        break;

SecActivity中接受注册并处理;

public class SecActivity extends AppCompatActivity {
    @BindView(R.id.sendData)
    Button mSendData;
    @BindView(R.id.receive)
    Button mReceive;
    @BindView(R.id.tv_receive)
    TextView mTvReceive;

    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_sec);
        ButterKnife.bind(this);
    }

    @OnClick({R.id.sendData, R.id.receive})
    public void onViewClicked(View view) {
        switch (view.getId()) {
            case R.id.sendData:
                //发送事件
                EventBus.getDefault().post(new UserEvent("Mr.sorrow", "123456"));
                finish();
                break;
            case R.id.receive:
                //要接收时开始注册
                EventBus.getDefault().register(SecActivity.this);
                break;
        }
    }

    //处理事件逻辑
    @Subscribe(threadMode = ThreadMode.MAIN, sticky = true)
    public void receiveEventBus(MessageEvent messageEvent) {
        mTvReceive.setText(messageEvent.toString());
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        //解注册
        EventBus.getDefault().removeAllStickyEvents();
        EventBus.getDefault().unregister(SecActivity.this);
    }
}

现在说一下RxBus,这是官网地址:

https://github.com/AndroidKnife/RxBus

跟EventBus相比,RxBus主要没有发送粘性事件的功能。

由于我想在我的项目用用事件管理器踢掉Intent传递事件。而我的项目中用了RxJava,本打算要用RxBus,由于没该功能,只能选用EventBus了。



  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 6
    评论
提供的源码资源涵盖了Java应用等多个领域,每个领域都包含了丰富的实例和项目。这些源码都是基于各自平台的最新技术和标准编写,确保了在对应环境下能够无缝运行。同时,源码中配备了详细的注释和文档,帮助用户快速理解代码结构和实现逻辑。 适用人群: 适合毕业设计、课程设计作业。这些源码资源特别适合大学生群体。无论你是计算机相关专业的学生,还是对其他领域编程感兴趣的学生,这些资源都能为你提供宝贵的学习和实践机会。通过学习和运行这些源码,你可以掌握各平台开发的基础知识,提升编程能力和项目实战经验。 使用场景及目标: 在学习阶段,你可以利用这些源码资源进行课程实践、课外项目或毕业设计。通过分析和运行源码,你将深入了解各平台开发的技术细节和最佳实践,逐步培养起自己的项目开发和问题解决能力。此外,在求职或创业过程中,具备跨平台开发能力的大学生将更具竞争力。 其他说明: 为了确保源码资源的可运行性和易用性,特别注意了以下几点:首先,每份源码都提供了详细的运行环境和依赖说明,确保用户能够轻松搭建起开发环境;其次,源码中的注释和文档都非常完善,方便用户快速上手和理解代码;最后,我会定期更新这些源码资源,以适应各平台技术的最新发展和市场需求。 所有源码均经过严格测试,可以直接运行,可以放心下载使用。有任何使用问题欢迎随时与博主沟通,第一时间进行解答!

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 6
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值