零、前言
让数据坐上bus,自由穿梭,效果如下: 通常我们可以使用Intent来实现,但通过EventBus更方便些
event_bus.gif
一、简介:
1.引入:
代码语言:javascript复制implementation 'org.greenrobot:eventbus:3.1.1'
代码语言:javascript复制Event 事件。它可以是任意类型。
Subscriber 事件订阅者。
Publisher 事件的发布者。
2.EventBus3.0有四种线程模型:后面具体分析
代码语言:javascript复制POSTING (默认)
表示事件处理函数的线程跟发布事件的线程在同一个线程。
MAIN
表示事件处理函数的线程在主线程(UI)线程,因此在这里不能进行耗时操作。
BACKGROUND
表示事件处理函数的线程在后台线程,因此不能进行UI操作。
如果发布事件的线程是主线程(UI线程),那么事件处理函数将会开启一个后台线程,
如果果发布事件的线程是在后台线程,那么事件处理函数就使用该线程。
ASYNC
表示无论事件发布的线程是哪一个,事件处理函数始终会新建一个子线程运行,同样不能进行UI操作。
二、简单使用
1.MainActivity
代码语言:javascript复制public class MainActivity extends AppCompatActivity {
@BindView(R.id.textView)
TextView mTextView;
@BindView(R.id.id_btn_go)
Button mIdBtnGo;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
ButterKnife.bind(this);
//1、注册EventBus
EventBus.getDefault().register(this);
System.out.println(Thread.currentThread().getName());
}
//3.Subscriber事件订阅者
@Subscribe(threadMode = ThreadMode.MAIN)
public void Event(Article article) {
mTextView.setText("作者 :" article.getAuthor() "n" "文章名字:《" article.getName() "》");
System.out.println(Thread.currentThread().getName());
}
@Override
protected void onDestroy() {
super.onDestroy();
//2.注销事件
if (EventBus.getDefault().isRegistered(this)) {
EventBus.getDefault().unregister(this);
}
}
@OnClick(R.id.id_btn_go)
public void onViewClicked() {
startActivity(new Intent(MainActivity.this, SecondActivity.class));
}
}
SecondActivity
代码语言:javascript复制public class SecondActivity extends AppCompatActivity {
@BindView(R.id.id_btn_back)
Button mIdBtnBack;
@BindView(R.id.id_et_content)
EditText mIdEtContent;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_sencond);
ButterKnife.bind(this);
}
@OnClick(R.id.id_btn_back)
public void onViewClicked() {
String txt = mIdEtContent.getText().toString();
Article toly = new Article(txt, "张风捷特烈");
//4.发布事件
EventBus.getDefault().post(toly);
System.out.println(Thread.currentThread().getName());
finish();
}
}
事件实体类
代码语言:javascript复制public class Article {
private String name;
private String author;
//get()、set()略。。。
}
测试1:POSTING与MAIN
SecondActivity中,新开线程发布事件
代码语言:javascript复制new Thread() {
@Override
public void run() {
String txt = mIdEtContent.getText().toString();
Article toly = new Article(txt, "张风捷特烈");
//4.发布事件
EventBus.getDefault().post(toly);
System.out.println(Thread.currentThread().getName());
finish();
}
}.start();
POSTING:崩了 Only the original thread that created a view hierarchy can touch its views. MAIN:正常显示
说明使用POSTING,发布与订阅在同一个线程,也就是子线程,更新UI会崩 说明使用MAIN,不管发布者在哪,订阅者都在main线程,可更新UI,但不能耗时操作
测试2:BACKGROUND
代码语言:javascript复制在主线程发布:看订阅执行函数 结果BACKGROUND崩了,说明订阅执行函数是新建一个线程中进行的
发布:main
订阅:pool-1-thread-1
代码语言:javascript复制如果在子线程发布:看订阅执行函数 则订阅执行函数在该子线程(通过线程id查看)
发布:Thread-5
订阅:Thread-5
测试3:ASYNC
代码语言:javascript复制在主线程发布:看订阅执行函数
I/System.out: 发布main
I/System.out: 订阅:pool-1-thread-1
代码语言:javascript复制如果在子线程发布:看订阅执行函数
I/System.out: 发布Thread-5
I/System.out: 订阅:pool-1-thread-1
他们之间的不同应该非常明显了吧。
后记、
1.声明:
[1]本文由张风捷特烈原创,转载请注明 [2]欢迎广大编程爱好者共同交流 [3]个人能力有限,如有不正之处欢迎大家批评指证,必定虚心改正 [4]你的喜欢与支持将是我最大的动力