0%

EventBus 3.0 相见恨晚

为什么要用EventBus?它是干什么用的?##

EventBus是什么###

EventBus是一个Android端优化的publish/subscribe消息总线,简化了应用程序内各组件间、组件与后台线程间的通信。

这样一个简单的概念,可能让你不足以感受到EventBus的魅力。我们来思考一下下面这些情况。

  • 使用购物类APP时,我们在商品详情页将商品加入购物车时,底部tab栏的购物车tab右上角的小数字也同步发生了变化。

面对这样的需求,一般情况下都是在详情页中提供一个接口,在tab栏所在的Activity(或者是Fragment)中注册这个接口,当点击事件发生是,回调这个接口,更新tab栏的内容。甚至有时候,由于Activity+多个fragment 的组合,可能需要多个接口经过层层传递才能实现某些在产品经理看来很简单的UI更新。

  • 大部分APP在用户完成用户登录操作后,需要在返回界面同步更新用户信息。

这个时候,我们一般会在返回界面的onActivityResult方法中,通过requestCode及resultCode做出种种判断,最后在确认用户登录成功的情况下,请求用户信息完成UI的更新,这种体验其实是非常不好,作为一个用户希望的是登录成功的同时完成用户信息的更新。

当然上面两种情况,用BroadcastReceiver实现,也是完全可以的,肯能会相对简单一些,但是从整体性能来说,这样是不好的。

而EventBus的出现,很好的解决了这些让我们头疼的问题。首先就用一个简单的列子来实现一下用户登录信息异步更新的情况。

EventBus 3.0 使用Demo##

这里用两个Activity简单的模拟常见的用户登录。

用户信息界面###

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
public class FirstActivity extends AppCompatActivity {
private Button btn;
private Context mContext;
//User Info
private TextView userName;
private ImageView usreImg;

@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
mContext = this;
EventBus.getDefault().register(mContext);
InitView();


}

private void InitView() {
setContentView(R.layout.activity_first);
userName = (TextView) findViewById(R.id.userName);
usreImg = (ImageView) findViewById(R.id.userImg);
btn = (Button) findViewById(R.id.button);
btn.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
startActivity(new Intent(mContext, SecondActivity.class));
}
});
}


@Subscribe
public void onUserEvent(UserEvent event) {
userName.setText("用户名:" + event.nameStr);
Glide.with(mContext).load(event.imgUrl).into(usreImg);

}

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

用户登录界面###

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
public class SecondActivity extends AppCompatActivity {
private Context mContext;
private Button btn;
private EditText name;

@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
mContext=this;
setContentView(R.layout.activity_second);
name = (EditText) findViewById(R.id.name);
findViewById(R.id.button).setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
//测试数据
String userImg = "http://img5.imgtn.bdimg.com/it/u=3774008831,1558972780&fm=21&gp=0.jpg";
//这里只是为了方便测试,随便写的登录逻辑
String userName="";
if(!TextUtils.isEmpty(name.getText().toString())){
userName = name.getText().toString();
UserEvent userEvent = new UserEvent(userName, userImg);
EventBus.getDefault().post(userEvent);
}else {
//不再发送事件Event
Toast.makeText(mContext,"登录失败",Toast.LENGTH_SHORT).show();
}

finish();
}
});

}
}

UserEvent 类###

1
2
3
4
5
6
7
8
9
public class UserEvent {
public final String nameStr;
public final String imgUrl;

public UserEvent(String nameStr, String imgUrl) {
this.nameStr = nameStr;
this.imgUrl = imgUrl;
}
}

这里可以看到没有Handler,没有onActivityResult的判断,没有没有任何接口。就实现了如gif图中所示效果。

接下来,就总结一下如何使用EventBus3.0 。

EventBus 3.0 使用步骤##

1.首先我们需要将EventBus添加到我们的项目中。在AndroidStudio中我们可以在gradle里面直接配置即可。

1
compile 'org.greenrobot:eventbus:3.0.0'

2.创建一个事件类(这个类似于JavaBean),就是上面的UserEvent类。

这里的Demo只是举例说明简单用法,实际中可以根据需要创建不同的事件类

3.注册

1
EventBus.getDefault().register(mContext);

4.订阅(响应事件方法)

1
2
3
4
5
6
@Subscribe
public void onUserEvent(UserEvent event) {
userName.setText("用户名:" + event.nameStr);
Glide.with(mContext).load(event.imgUrl).into(usreImg);

}

这里的注解@Subscribe 很关键,表明这个方法为订阅者,这个方法的名字也已经不在重要了(相对于以前的版本来说),在这个方法里,我们实现了UI更新。

5.分发事件

1
2
UserEvent userEvent = new UserEvent(userName, userImg);
EventBus.getDefault().post(userEvent);

6.解除注册

1
EventBus.getDefault().unregister(mContext);
加个鸡腿呗.