手写EventBus框架——动手_整体架构设计

路漫漫其修远兮

01. 手写EventBus框架——源码分析1
02. 手写EventBus框架——源码分析2
03. 手写EventBus框架——动手_整体架构设计
04. 手写EventBus框架——动手_终结


o.O前戏已经足够了,接下来撸     代码了。

主体框架

类设计

第一版结构

EventBus.class

/**
 * <p/>
 * Author: 温利东 on 2017/3/26 10:44.
 * blog: http://www.jianshu.com/u/99f514ea81b3
 * github: https://github.com/LidongWen
 */

public class EventBus {
    private static EventBus ourInstance;

    private final Map<Class<?>, CopyOnWriteArrayList<Subscription>> cacheMap;
    private final Map<Object, List<Class<?>>> typesBySubscriber;
    //粘性
    private final Map<Class<?>, Object> stickyEvents;

    public static EventBus getDefault() {
        if (ourInstance == null) {
            synchronized (EventBus.class) {
                if (ourInstance == null) {
                    ourInstance = new EventBus();
                }
            }
        }
        return ourInstance;
    }

    private EventBus() {
        //初始化一些数据
        cacheMap = new HashMap<>();
        stickyEvents = new ConcurrentHashMap<>();
        typesBySubscriber=new HashMap<>();
    }

    public void register(Object subscriber) {
        //1.找到所有方法
        //2.保存订阅  subscriber();
    }

    public void unRegister(Object subscriber) {
        //去除订阅
    }
    public void post(Object obj) {
        //1.获取缓存 进行判断
        //2.指定线程执行
    }
    public void postSticky(Object obj){
        //加入粘性缓存 stickyEvents
        //执行
    }

    /**
     *  保存订阅方法
     * @param subscriber
     */
    private void subscribe(Object subscriber, SubscriberMethod subscriberMethod) {
        //1.保存数据  ,  如果重复 抛异常
        //2.执行粘性事件
    }
}

线程枚举类

public enum ThreadMode {

    POSTING,

    MAIN,

    BACKGROUND,

    ASYNC
}

注解类

@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.METHOD})
public @interface Subscribe {
    ThreadMode threadMode() default ThreadMode.POSTING;
    //粘性
    boolean sticky() default false;
    //优先级
    int priority() default 0;
}

订阅元元素类 SubscriberMethod ;

/**
 * <p/>
 * Author: 温利东 on 2017/3/23 17:56.
 * blog: http://blog.csdn.net/sinat_15877283
 * github: https://github.com/LidongWen
 */

public class SubscriberMethod {
    final Method method;
    final ThreadMode threadMode;
    final Class<?> eventType;       //参数
    final int priority;
    final boolean sticky;
    /**
     * Used for efficient comparison
     */
    String methodString;

    public SubscriberMethod( Method method, Class<?> eventType, ThreadMode threadMode, int priority, boolean sticky) {
        this.method = method;
        this.threadMode = threadMode;
        this.eventType = eventType;
        this.priority = priority;
        this.sticky = sticky;
    }

    @Override
    public boolean equals(Object other) {
        if (other == this) {
            return true;
        } else if (other instanceof SubscriberMethod) {
            checkMethodString();
            SubscriberMethod otherSubscriberMethod = (SubscriberMethod) other;
            otherSubscriberMethod.checkMethodString();
            // Don't use method.equals because of http://code.google.com/p/android/issues/detail?id=7811#c6
            return methodString.equals(otherSubscriberMethod.methodString);
        } else {
            return false;
        }
    }

    private synchronized void checkMethodString() {
        if (methodString == null) {
            // Method.toString has more overhead, just take relevant parts of the method
            StringBuilder builder = new StringBuilder(64);
            builder.append(method.getDeclaringClass().getName());
            builder.append('#').append(method.getName());
            builder.append('(').append(eventType.getName());
            methodString = builder.toString();
        }
    }

    @Override
    public int hashCode() {
        return method.hashCode();
    }
}

扩展框架设计_01

以上整体的一个架构就出来了,可以看出来,整体的逻辑都在EventBus类里面,虽然其中都是些伪代码,但代码量还是很大的,那么我们可以抽出一些伪代码作为帮助类;

  1. 订阅方法查找类;SubscriberMethodFinder.class;
  2. 方法执行帮助类; InvokeHelper.class;

那么结构就变成这样了


InvokeHelper.class

/**
 *  方法执行帮助类
 * <p/>
 * Author: 温利东 on 2017/3/26 11:29.
 * blog: http://www.jianshu.com/u/99f514ea81b3
 * github: https://github.com/LidongWen
 */
public class InvokeHelper {

    public static void post(SubscriberMethod subscriberMethod,Object event, boolean isMainThread) {
        switch (subscriberMethod.threadMode) {
            case POSTING:
                //直接执行
                break;
            case MAIN:
                if (isMainThread) {
                    //直接执行
                } else {
                    // 放在handler内执行
                }
                break;
            case BACKGROUND:
                if (isMainThread) {
                    //放在后台线程执行
                } else {
                    //执行
                }
                break;
            case ASYNC:
                //放在异步线程内执行
                break;
            default:
                //抛异常
                throw new IllegalStateException("Unknown thread mode: " + subscriberMethod.threadMode);
        }
    }
}

SubscriberMethodFinder.class

/**
 *  订阅方法查找类
 * <p/>
 * Author: 温利东 on 2017/3/26 11:21.
 * blog: http://www.jianshu.com/u/99f514ea81b3
 * github: https://github.com/LidongWen
 */
public class SubscriberMethodFinder {
    private static final Map<Class<?>, List<SubscriberMethod>> METHOD_CACHE = new ConcurrentHashMap<>();

    /**
     * 查找方法
     * @param subscriberClass
     * @return
     */
    List<SubscriberMethod> findSubscriberMethods(Class<?> subscriberClass) {
        return null;
    }
}

扩展框架设计_02

  1. 异步线程 AsyncPoster
  2. 顺序执行BackgroundPoster

这下子架构就变成这样了


/**
 * //异步线程
 * <p/>
 * Author: 温利东 on 2017/3/26 17:36.
 * blog: http://www.jianshu.com/u/99f514ea81b3
 * github: https://github.com/LidongWen
 */
public class AsyncPoster implements Runnable {
    @Override
    public void run() {

    }
}
/**
 * Posts events in background.
 * <p/>
 * Author: 温利东 on 2017/3/26 17:36.
 * blog: http://www.jianshu.com/u/99f514ea81b3
 * github: https://github.com/LidongWen
 */
final class BackgroundPoster implements Runnable {


    @Override
    public void run() {

    }

}

还漏了一个 Subscription.class 这里加上

Subscription.class

final class Subscription {
    final Object subscriber;
    final SubscriberMethod subscriberMethod;

    volatile boolean active;

    Subscription(Object subscriber, SubscriberMethod subscriberMethod) {
        this.subscriber = subscriber;
        this.subscriberMethod = subscriberMethod;
        active = true;
    }

    @Override
    public boolean equals(Object other) {
        if (other instanceof Subscription) {
            Subscription otherSubscription = (Subscription) other;
            return subscriber == otherSubscription.subscriber
                    && subscriberMethod.equals(otherSubscription.subscriberMethod);
        } else {
            return false;
        }
    }

    @Override
    public int hashCode() {
        return subscriber.hashCode() + subscriberMethod.methodString.hashCode();
    }
}

好了 现在我们简易版EventBus框架结构就搭建起来了。其中没有实现代码,全是伪代码逻辑。其实写到这里,各位都可以自己敲代码来实现具体的功能了。

下一节我们将完成以及完善我们简易版EventBus的功能了。

下一节: 04. 手写EventBus框架——动手_终结


希望我的文章不会误导在观看的你,如果有异议的地方欢迎讨论和指正。
如果能给观看的你带来收获,那就是最好不过了。

人生得意须尽欢, 桃花坞里桃花庵
点个关注呗,对,不信你点试试?
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容