为什么要用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简单的模拟常见的用户登录。效果图如下
用户信息界面###
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();
}
}
用户登录界面###
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://img2.imgtn.bdimg.com/it/u=262221958,4109901128&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 类###
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里面直接配置即可。
compile 'org.greenrobot:eventbus:3.0.0'
2.创建一个事件类(这个类似于JavaBean),就是上面的UserEvent类。
这里的Demo只是举例说明简单用法,实际中可以根据需要创建不同的事件类
3.注册
EventBus.getDefault().register(mContext);
4.订阅(响应事件方法)
@Subscribe
public void onUserEvent(UserEvent event) {
userName.setText("用户名:" + event.nameStr);
Glide.with(mContext).load(event.imgUrl).into(usreImg);
}
这里的注解@Subscribe 很关键,表明这个方法为订阅者,这个方法的名字也已经不在重要了(相对于以前的版本来说),在这个方法里,我们实现了UI更新,将用户信息更新出来。
5.分发事件
UserEvent userEvent = new UserEvent(userName, userImg);
EventBus.getDefault().post(userEvent);
6.解除注册
EventBus.getDefault().unregister(mContext);
这里仅从一个最简单的Demo,了解了一下EventBus是多么的神奇。此处没有网络请求,post方法也是在主线程中,所以默认情况下相应事件方法onUserEvent也会在主线程中执行。实际上onUserEvent方法在注解中还是需要添加参数的。
好了,这里就简单了解一下EventBus3.0 ,下篇文章就其使用方法深入了解一下。