Android EventBus 3.0 相见恨晚

719457222 8年前
   <h2>为什么要用EventBus?它是干什么用的?</h2>    <h3>EventBus是什么</h3>    <p>EventBus是一个Android端优化的publish/subscribe消息总线,简化了应用程序内各组件间、组件与后台线程间的通信。</p>    <p>这样一个简单的概念,可能让你不足以感受到EventBus的魅力。我们来思考一下下面这些情况。</p>    <blockquote>     <p>使用购物类APP时,我们在商品详情页将商品加入购物车时,底部tab栏的购物车tab右上角的小数字也同步发生了变化。</p>    </blockquote>    <p>面对这样的需求,一般情况下都是在详情页中提供一个接口,在tab栏所在的Activity(或者是Fragment)中注册这个接口,当点击事件发生是,回调这个接口,更新tab栏的内容。甚至有时候,由于我们的应用会是Activity+多个fragment 的组合,可能需要多个接口经过层层传递,才能实现某些在产品经理看来很简单的UI更新。</p>    <blockquote>     <p>大部分APP在用户完成用户登录操作后,需要在返回界面同步更新用户信息。</p>    </blockquote>    <p>这个时候,我们一般会在返回界面的onActivityResult方法中,通过requestCode及resultCode做出种种判断,最后在确认用户登录成功的情况下,请求用户信息完成UI的更新,这种体验其实是非常不好,作为一个用户希望的是,登录成功的同时完成用户信息的更新。</p>    <p>当然上面两种情况,用BroadcastReceiver实现,也是完全可以的,肯能会相对简单一些,但是从整体性能来说,这样是不好的;再者古人云,杀鸡焉用牛刀!!!</p>    <p>而EventBus的出现,很好的解决了这些让我们头疼的问题。首先就用一个简单的列子来实现一下用户登录信息<strong>异步</strong>更新的情况。</p>    <h2>EventBus 3.0 使用Demo</h2>    <p>这里用两个Activity简单的模拟常见的用户登录。效果图如下</p>    <p><img src="https://simg.open-open.com/show/f610ed0f8ab5708099574770e90d68cf.gif"></p>    <p>登录</p>    <h3>用户信息界面</h3>    <pre>  <code class="language-java">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();      }  }</code></pre>    <h3>用户登录界面</h3>    <pre>  <code class="language-java">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();              }          });        }  }</code></pre>    <h3>UserEvent 类</h3>    <pre>  <code class="language-java">public class UserEvent {      public final String nameStr;      public final String imgUrl;        public UserEvent(String nameStr, String imgUrl) {          this.nameStr = nameStr;          this.imgUrl = imgUrl;      }  }</code></pre>    <p>这里可以看到没有Handler,没有onActivityResult的判断,没有没有任何接口。就实现了如gif图中所示效果。</p>    <p>接下来,就总结一下如何使用EventBus3.0 。</p>    <h2>EventBus 3.0 使用步骤</h2>    <p>1.首先我们需要将EventBus添加到我们的项目中。在AndroidStudio中我们可以在gradle里面直接配置即可。</p>    <pre>  <code class="language-java">compile 'org.greenrobot:eventbus:3.0.0'</code></pre>    <p>2.创建一个事件类(这个类似于JavaBean),就是上面的UserEvent类。</p>    <p>这里的Demo只是举例说明简单用法,实际中可以根据需要创建不同的事件类</p>    <p>3.注册</p>    <pre>  <code class="language-java">  EventBus.getDefault().register(mContext);</code></pre>    <p>4.订阅(响应事件方法)</p>    <pre>  <code class="language-java">@Subscribe      public void onUserEvent(UserEvent event) {          userName.setText("用户名:" + event.nameStr);          Glide.with(mContext).load(event.imgUrl).into(usreImg);        }</code></pre>    <p>这里的注解@Subscribe 很关键,表明这个方法为订阅者,这个方法的名字也已经不在重要了(相对于以前的版本来说),在这个方法里,我们实现了UI更新,将用户信息更新出来。</p>    <p>5.分发事件</p>    <pre>  <code class="language-java">UserEvent userEvent = new UserEvent(userName, userImg);                      EventBus.getDefault().post(userEvent);</code></pre>    <p>6.解除注册</p>    <pre>  <code class="language-java">EventBus.getDefault().unregister(mContext);</code></pre>    <blockquote>     <p>这里仅从一个最简单的Demo,了解了一下EventBus是多么的神奇。此处没有网络请求,post方法也是在主线程中,所以默认情况下相应事件方法onUserEvent也会在主线程中执行。实际上onUserEvent方法在注解中还是需要添加参数的。</p>    </blockquote>    <p>好了,这里就简单了解一下EventBus3.0 ,下篇文章就其使用方法深入了解一下。</p>    <p><br> 来自:http://www.jianshu.com/p/be963f80f309<br>  </p>    <p> </p>