Android 框架炼成 教你如何写组件间通信框架EventBus,2024年是做安卓开发人员的绝佳时机

关于Eventbus的介绍,前面已经有两篇:Android EventBus实战 没听过你就out了Android EventBus源码解析 带你深入理解EventBus , 如果你觉得还有问题,没关系,接下来我带大家手把手打造从无到有的编写这样的框架~~~

首先我们回顾一下,这玩意就是在register时,扫描类中复合命名规范的方法,存到一个map,然后post的时候,查找到匹配的方法,反射调用;好,那么根据这一句话,我们就开始编写框架之旅~~~

2、依然是原来的配方

==========

以下出现的实例代码和Android EventBus实战 没听过你就out了基本一致,所以我就贴出部分

1、ItemListFragment


package com.angeldevil.eventbusdemo;

import android.os.Bundle;

import android.support.v4.app.ListFragment;

import android.view.View;

import android.widget.ArrayAdapter;

import android.widget.ListView;

import com.angeldevil.eventbusdemo.Event.ItemListEvent;

import com.zhy.eventbus.EventBus;

public class ItemListFragment extends ListFragment

{

@Override

public void onCreate(Bundle savedInstanceState)

{

super.onCreate(savedInstanceState);

// Register

EventBus.getInstatnce().register(this);

}

@Override

public void onDestroy()

{

super.onDestroy();

// Unregister

EventBus.getInstatnce().unregister(this);

}

@Override

public void onViewCreated(View view, Bundle savedInstanceState)

{

super.onViewCreated(view, savedInstanceState);

// 开启线程加载列表

new Thread()

{

public void run()

{

try

{

Thread.sleep(2000); // 模拟延时

// 发布事件,在后台线程发的事件

EventBus.getInstatnce().post(new ItemListEvent(Item.ITEMS));

} catch (InterruptedException e)

{

e.printStackTrace();

}

};

}.start();

}

public void onEventUI(ItemListEvent event)

{

setListAdapter(new ArrayAdapter(getActivity(),

android.R.layout.simple_list_item_activated_1,

android.R.id.text1, event.getItems()));

}

@Override

public void onListItemClick(ListView listView, View view, int position,

long id)

{

super.onListItemClick(listView, view, position, id);

EventBus.getInstatnce().post(getListView().getItemAtPosition(position));

}

}

2、ItemDetailFragment


package com.angeldevil.eventbusdemo;

import android.os.Bundle;

import android.support.v4.app.Fragment;

import android.view.LayoutInflater;

import android.view.View;

import android.view.ViewGroup;

import android.widget.TextView;

import com.zhy.eventbus.EventBus;

public class ItemDetailFragment extends Fragment

{

private TextView tvDetail;

@Override

public void onCreate(Bundle savedInstanceState)

{

super.onCreate(savedInstanceState);

// register

EventBus.getInstatnce().register(this);

}

@Override

public void onDestroy()

{

super.onDestroy();

// Unregister

EventBus.getInstatnce().unregister(this);

}

/** List点击时会发送些事件,接收到事件后更新详情 */

public void onEventUI(Item item)

{

if (item != null)

tvDetail.setText(item.content);

}

@Override

public View onCreateView(LayoutInflater inflater, ViewGroup container,

Bundle savedInstanceState)

{

View rootView = inflater.inflate(R.layout.fragment_item_detail,

container, false);

tvDetail = (TextView) rootView.findViewById(R.id.item_detail);

return rootView;

}

}

可以看到,我们在ItemListFragment里面使用了:

EventBus.getInstatnce().post(new ItemListEvent(Item.ITEMS));去发布了一个事件,然后更新了我们的列表;

点击Item的时候,使用EventBus.getInstatnce().post(getListView().getItemAtPosition(position));发布了一个事件,更新了我们的ItemDetailFragment的列表;

效果:

效果图和之前的一摸一样~~~

但是请注意,现在我们用的是EventBus.getInstatnce();并发是EventBus.getDefault();并且看下包名import com.zhy.eventbus.EventBus;

我想你应该明白了,这是我们自己写的类来实现的~~~~

好了,接下来就带大家一起实现这个类~~

ps :以上代码和效果图,完全是为了博客的完整性,勿见怪~~

3、无中生有

======

1、getInstance


我们这里为了方便,直接简单粗暴的使用恶汉模式创建单例:

private static EventBus eventBus = new EventBus();

public static EventBus getInstatnce()

{

return eventBus;

}

private EventBus()

{

mHandler = new Handler(Looper.getMainLooper());

}

然后在构造方法中初始化了一个mHandler,没错,它就是用来在处理在UI线程调用方法的。

接下来看register

2、register


/*

  • 我们的强大的map,存储我们的方法

*/

private static Map<Class, CopyOnWriteArrayList> mSubscribeMethodsByEventType = new HashMap<Class, CopyOnWriteArrayList>();

public void register(Object subscriber)

{

Class clazz = subscriber.getClass();

Method[] methods = clazz.getDeclaredMethods();

CopyOnWriteArrayList subscribeMethods = null;

/**

  • 遍历所有方法

*/

for (Method method : methods)

{

String methodName = method.getName();

/**

  • 判断方法是否以onEvent的开头

*/

if (methodName.startsWith(“onEvent”))

{

SubscribeMethod subscribeMethod = null;

// 方法命中提前在什么线程运行。默认在UI线程

String threadMode = methodName.substring(“onEvent”.length());

ThreadMode mode = ThreadMode.UI;

Class<?>[] parameterTypes = method.getParameterTypes();

// 参数的个数为1

if (parameterTypes.length == 1)

{

Class<?> eventType = parameterTypes[0];

synchronized (this)

{

if (mSubscribeMethodsByEventType.containsKey(eventType))

{

subscribeMethods = mSubscribeMethodsByEventType

.get(eventType);

} else

{

subscribeMethods = new CopyOnWriteArrayList();

mSubscribeMethodsByEventType.put(eventType,

subscribeMethods);

}

}

if (threadMode.equals(“Async”))

{

mode = ThreadMode.Async;

}

// 提取出method,mode,方法所在类对象,存数的类型封装为SubscribeMethod

subscribeMethod = new SubscribeMethod(method, mode,

subscriber);

subscribeMethods.add(subscribeMethod);

}

}

}

}

enum ThreadMode

{

UI, Async

}

class SubscribeMethod

{

Method method;

ThreadMode threadMode;

Object subscriber;

public SubscribeMethod(Method method, ThreadMode threadMode,

Object subscriber)

{

this.method = method;

this.threadMode = threadMode;

this.subscriber = subscriber;

}

}

可以看到我们使用了一个Map存储所有的方法,key为参数的类型class;value为CopyOnWriteArrayList

这里我们封装了一个SubscribeMethod,这个里面存储了我们需要运行方法的所有参数,毕竟我们运行时,需要该方法,该方法所在的对象,以及在什么线程运行;三个对象足以,当然也缺一不可了~~

register里面,我们遍历该类的所有方法,找到onEvent开头的,封装成SubscribeMethod,存在Map里面,当然了,一个参数类型对应很多方法,所以value是个CopyOnWriteArrayList。

扫描完成,我们就完成了将方法的存储。

还有一点,我们这里默认在UI线程执行,如果方法是onEventAsync则认为在子线程执行,我们也只支持这两种模式,简化一点~

3、post


private static ThreadLocal mPostingThread = new ThreadLocal()

{

@Override

public PostingThread get()

{

return new PostingThread();

}

};

public void post(Object eventTypeInstance)

{

//拿到该线程中的PostingThread对象

PostingThread postingThread = mPostingThread.get();

postingThread.isMainThread = Looper.getMainLooper() == Looper

.myLooper();

//将事件加入事件队列

List eventQueue = postingThread.mEventQueue;

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
img
img
img
img
img
img
img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录大纲截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且后续会持续更新

如果你觉得这些内容对你有帮助,可以添加V获取:vip204888 (备注Android)
img

题外话

不管怎么样,不论是什么样的大小面试,要想不被面试官虐的不要不要的,只有刷爆面试题题做好全面的准备,当然除了这个还需要在平时把自己的基础打扎实,这样不论面试官怎么样一个知识点里往死里凿,你也能应付如流啊

这里我为大家准备了一些我工作以来以及参与过的大大小小的面试收集总结出来的一套进阶学习的视频及面试专题资料包,主要还是希望大家在如今大环境不好的情况下面试能够顺利一点,希望可以帮助到大家~

欢迎评论区讨论。

本文已被CODING开源项目:《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》收录

一个人可以走的很快,但一群人才能走的更远。如果你从事以下工作或对以下感兴趣,欢迎戳这里加入程序员的圈子,让我们一起学习成长!

AI人工智能、Android移动开发、AIGC大模型、C C#、Go语言、Java、Linux运维、云计算、MySQL、PMP、网络安全、Python爬虫、UE5、UI设计、Unity3D、Web前端开发、产品经理、车载开发、大数据、鸿蒙、计算机网络、嵌入式物联网、软件测试、数据结构与算法、音视频开发、Flutter、IOS开发、PHP开发、.NET、安卓逆向、云计算

么样,不论是什么样的大小面试,要想不被面试官虐的不要不要的,只有刷爆面试题题做好全面的准备,当然除了这个还需要在平时把自己的基础打扎实,这样不论面试官怎么样一个知识点里往死里凿,你也能应付如流啊**

这里我为大家准备了一些我工作以来以及参与过的大大小小的面试收集总结出来的一套进阶学习的视频及面试专题资料包,主要还是希望大家在如今大环境不好的情况下面试能够顺利一点,希望可以帮助到大家~

[外链图片转存中…(img-XLqq1sMy-1712148157602)]

欢迎评论区讨论。

本文已被CODING开源项目:《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》收录

一个人可以走的很快,但一群人才能走的更远。如果你从事以下工作或对以下感兴趣,欢迎戳这里加入程序员的圈子,让我们一起学习成长!

AI人工智能、Android移动开发、AIGC大模型、C C#、Go语言、Java、Linux运维、云计算、MySQL、PMP、网络安全、Python爬虫、UE5、UI设计、Unity3D、Web前端开发、产品经理、车载开发、大数据、鸿蒙、计算机网络、嵌入式物联网、软件测试、数据结构与算法、音视频开发、Flutter、IOS开发、PHP开发、.NET、安卓逆向、云计算

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值