源码 github.com/harvie1208/…git
关键词:观察者模式、反射、自定义注解、线程调度github
手写200行代码,一步一步实现EventBus核心功能,看完能够写一套属于本身的事件总线库啦!数组
不知你们日常在看博客的时候有没有和我遇到同样的问题,就是看的是懂非懂,好像懂了,又好像没懂。bash
主要有如下两点:框架
在求知的路上,我也看了很多文章,有很是优秀的,也有缺这少那的。一路走来填了很多坑,后面我会将所学知识点整理出来分享给你们,尽可能作到通俗易懂的理论加完整案例源码。一方面是对本身知识点的总结回顾,另外一方面也但愿能帮助到有须要的同窗少走弯路。因技术水平有限,若有不正之处,还望各位不吝指教。异步
EventBus顾名思义就是事件总线,实际上就是一个事件发布者/事件监听者(订阅者)
的框架, 发布者发布事件,Bus自动处理与分发,监听者被动的接受。简化各类异步和跳转的通讯。oop
1.短信验证码登录场景post
主登录界面A->输入手机号界面B->短信验证码界面C->登录成功跳转首页D
需求:登录成功后须要关闭A、B、C三个页面
复制代码
2.音乐播放场景ui
假如首页有5个tab(包含5个fragment),每一个fragment中都有音乐播放状态小图标
需求:音乐播放或暂停时,须要更新全部播放状态图标
复制代码
使用观察者模式,在须要接收事件的方法上添加订阅注解标识,并将此方法所在对象添加到订阅者集合,
发送事件时遍历订阅者集合,在经过反射调用相关订阅方法。
复制代码
public class EventBus {
private static EventBus myBus;
public static EventBus getInstance(){
if (myBus==null){
synchronized (EventBus.class){
if (myBus==null){
myBus = new EventBus();
}
}
}
return myBus;
}
}
复制代码
建立自定义注解@Subscribe用来标示订阅方法spa
注解Annontation是Java5开始引入的新特征,通俗来讲就是为程序的元素(类、方法、成员变量)添加标记用的
@Target(ElementType.METHOD) //表示此注解做用域在方法上
@Retention(RetentionPolicy.RUNTIME) //编译程序处理完注解信息后存储在class中,可由VM读入
public @interface Subscribe {
ThreadModel thread();//用于指定被注解方法执行时所在的线程
}
复制代码
使用注解
public class MainActivity extends AppCompatActivity {
@Subscribe(thread = ThreadModel.BACKGROUND)//指定在子线程中执行
public void haha(LoginEvent loginEvent){
Log.e("EventBus",loginEvent.getLoginStatus()+Thread.currentThread().getName());
}
}
复制代码
先声明一个集合用于存储类对象和被注解的方法及线程模式
遍历注册对象的全部方法,取出带有@Subscribe注解的方法
获取参数类型数组,当前仅支持一个参数
获取指定线程模式
构建订阅者实例(方法、参数类型、线程模式),加入订阅集合
public class EventBus {
//存储订阅类及方法参数
private Map<Object,List<Subscriber>> subscribeMethod;
public void register(Object obj){
if (obj==null){
return;
}
Class<?> mclazz = obj.getClass();
//获取本类全部方法
Method[] methods = mclazz.getDeclaredMethods();
List<Subscriber> methods1 = new ArrayList<>();
for (Method method : methods){
//获取带有咱们Subscribe注解的方法
Subscribe subscribe = method.getAnnotation(Subscribe.class);
if (subscribe==null){
continue;
}
//获取参数类型集合
Class<?>[] typeVariable = method.getParameterTypes();
if (typeVariable.length!=1){
continue;
}
ThreadModel threadModel = subscribe.thread();
Subscriber busMethod = new Subscriber(method,threadModel,typeVariable[0]);
methods1.add(busMethod);
}
if (methods1.size()>0){
subscribeMethod.put(obj,methods1);
}
}
}
复制代码
public void unRegister(Object object){
if (subscribeMethod.containsKey(object)){
subscribeMethod.remove(object);
}
}
复制代码
根据发送事件参数类型,遍历集合找到对应方法
判断线程模式,主线程用handler处理,子线程用线程池处理
反射调用方法将事件传过去
public class EventBus {
//存储订阅类及方法参数
private Map<Object,List<Subscriber>> subscribeMethod;
//线程调度
private Handler mHandler;
//线程池
private ExecutorService executorService;
private EventBus(){
subscribeMethod = new HashMap<>();
mHandler = new Handler(Looper.getMainLooper());
executorService = Executors.newCachedThreadPool();
}
public void postEvent(Object eventParam){
Set<Object> set = subscribeMethod.keySet();
Iterator<Object> iterable =set.iterator();
while (iterable.hasNext()){
Object obj = iterable.next();
List<Subscriber> busMethodList = subscribeMethod.get(obj);
for (Subscriber busMethod : busMethodList){
if(busMethod.getParamsType() == eventParam.getClass()){
invoke(obj,busMethod,eventParam);
}
}
}
}
private void invoke(final Object obj, final Subscriber busMethod, final Object eventParam){
switch (busMethod.getThreadModel()){
case MAIN:
//经过handler调度到主线程
mHandler.post(new EventRunable(busMethod, obj, eventParam));
break;
default:
//交由线程池处理
executorService.execute(new EventRunable(busMethod, obj, eventParam));
break;
}
}
}
复制代码
事件参数与接收参数类型一致便可,方法名随意
EventBus.getInstance().postEvent(new LoginEvent("登陆成功"));
复制代码
不少看似高大上的框架其实也没咱们想的那么难,写着写着就顺手了,知而不行为不知,快动起手来吧!