Android插件化-Broadcast篇,2024年最新安卓面试自我介绍

简介: Android插件化-Broadcast篇,2024年最新安卓面试自我介绍
public static abstract class Component {
public final ArrayList intents;
public final String className;
public Bundle metaData;
public Package owner;
/** The order of this component in relation to its peers */
public int order;
ComponentName componentName;
String componentShortName;
//其他方法实现

注意对于静态注册在插件AndroidManifest.xml文件中的Receiver,宿主需要知道Receiver的className以及其对应注册的Action,这样在宿主中才能实现广播的中转;因此整个解析过程也就需要解析出广播的ClassName以及其注册的Action就OK了。

对于ClassName在Component已经出现,而对于Action我们则继续看Component类中的intents这个列表,该列表中所存储的对象对应的类集成至IntentInfo,因此我们还需要继续往上看才行。

public static abstract class IntentInfo extends IntentFilter {
public boolean hasDefault;
// 其他属性以及方法

而该内部类又继承至类IntentFilter,这里IntentFilter则不再是PackageParser类的内部类了,它就是我们在动态注册Reciver所用到的IntentFilter;而在IntentFilter类中我们则能够直接通过其mActions属性拿到当前ClasName对应Receiver所静态注册的Action了。

2、插件中静态注册广播解析实现

接下来就是通过反射的方式调用PackageParser的parsePackage获取到PackageParser$Package对象,接着就是一步一步的对该对象进行解析了。对应源码如下:

public static void parsePackage(String apkPath) {
try {
//根据插件本地存储的文件地址生成对应的文件
File file = new File(apkPath);
if (!file.exists()) {
Log.i(TAG, “parse plugin receiver apk not exist”);
return;
}
Log.i(TAG, "parse package path is " + apkPath);
//获取到PackageParser类对象
Class cls = Class.forName(“android.content.pm.PackageParser”);
Object packageParserObj = RefInvoke.createObject(cls, null, null);
if (null == packageParserObj) {
Log.i(TAG, “parse package create packageParser object failed”);
return;
}
//调用parsePackage方法
Object packageObj = RefInvoke.on(packageParserObj, “parsePackage”, new Class[]{File.class, int.class})
.invoke(file, PackageManager.GET_RECEIVERS);
if (null == packageObj) {
Log.i(TAG, “parse package get packageObj failed”);
return;
}
//获取PackageParser$Package对象中的receivers列表
List receivers = (List) RefInvoke.getFieldValue(RefInvoke.getField(packageObj.getClass(), “receivers”), packageObj);
if (null == receivers) {
Log.i(TAG, “parse package get receivers failed”);
return;
}
//遍历receivers列表获取AndroidManifest.xml文件中所注册的Receiver信息
for (Object receiver : receivers) {
parseAction(receiver, apkPath);
}
} catch (Exception e) {
Log.i(TAG, "parse package failed " + e);
}
}

上面代码就是通过反射的方式调用PackageParser的parsePackage方法,并遍历获取到的receivers列表。接下来就就是对receivers列表中的每个Receiver相关信息进行解析了。

private static void parseAction(Object receiver, String path) {
try {
//根据反射获取到PackageParserC o m p o n e n t 对象中的 i n t e n t s 列表 C l a s s < ? > c l s = R e f I n v o k e . g e t C l a s s ( " a n d r o i d . c o n t e n t . p m . P a c k a g e P a r s e r Component对象中的intents列表 Class cls = RefInvoke.getClass("android.content.pm.PackageParserComponent对象中的intents列表Class<?>cls=RefInvoke.getClass("android.content.pm.PackageParserComponent");
ArrayList intents = (ArrayList) RefInvoke.getFieldValue(RefInvoke.getField(cls, “intents”), receiver);
if (null == intents || 0 == intents.size()) {
return;
}
//获取该Receiver对应的ClassName
String clsName = (String) RefInvoke.getFieldValue(RefInvoke.getField(cls, “className”), receiver);
Log.i(TAG, "parseAction current receiver name is " + clsName);
//根据插件所在文件地址生成对应的ClassLoader并根据获取到的ClassName生成对应的对象
Object receiverObj = creatReceiverObj(clsName, path);
if (null == receiverObj) {
Log.i(TAG, “parseAction create receiver obj failed”);
return;
}
//接着就是遍历该Receiver中所注册的Action并存储在内存中以便宿主在接收到对应Action的广播之后能够直接进行转发
for (IntentFilter intentFilter : intents) {
Class intentFilterCls = RefInvoke.getClass(“android.content.IntentFilter”);
Log.i(TAG, "field: " + RefInvoke.getField(intentFilterCls, “mActions”));
List actions = (List) RefInvoke.getFieldValue(RefInvoke.getField(intentFilterCls, “mActions”), intentFilter);
for (String action : actions) {
registerActionToReceiver(action, receiverObj);
}
}
} catch (Exception e) {
Log.i(TAG, "parseAction failed " + e);
}
}
private static Object creatReceiverObj(String clsName, String path) {
try {
DexClassLoader dexClassLoader = DeHostDexClassloader.getInstance().getDexClassLoader(DePluginApplication.getContext(), path);
Class cls = dexClassLoader.loadClass(clsName);
return RefInvoke.createObject(cls, null, null);
} catch (Exception e) {
Log.i(TAG, "createReceiverObj failed " + e.getCause());
}
return null;
}


相关文章
|
7天前
|
数据管理 API 数据库
探索Android Jetpack:现代安卓开发的利器
Android Jetpack是谷歌为简化和优化安卓应用开发而推出的一套高级组件库。本文深入探讨了Jetpack的主要构成及其在应用开发中的实际运用,展示了如何通过使用这些工具来提升开发效率和应用性能。
|
1天前
|
安全 Java 编译器
Android面试题之Java 泛型和Kotlin泛型
**Java泛型是JDK5引入的特性,用于编译时类型检查和安全。泛型擦除会在运行时移除类型参数,用Object或边界类型替换。这导致几个限制:不能直接创建泛型实例,不能使用instanceof,泛型数组与协变冲突,以及在静态上下文中的限制。通配符如<?>用于增强灵活性,<? extends T>只读,<? super T>只写。面试题涉及泛型原理和擦除机制。
13 3
Android面试题之Java 泛型和Kotlin泛型
|
2天前
|
人工智能 安全 物联网
【Android】安卓开发的前景
【Android】安卓开发的前景
11 1
|
2天前
|
网络协议 算法 安全
小米安卓春招面试一面
小米安卓春招面试一面
20 3
|
2天前
|
缓存 网络协议 Java
Android面试题之Java网络通信基础知识
Socket是应用与TCP/IP通信的接口,封装了底层细节。网络通信涉及连接、读写数据。BIO是同步阻塞,NIO支持多路复用(如Selector),AIO在某些平台提供异步非阻塞服务。BIO示例中,服务端用固定线程池处理客户端请求,客户端发起连接并读写数据。NIO的关键是Selector监控多个通道的事件,减少线程消耗。书中推荐《Java网络编程》和《UNIX网络编程》。关注公众号AntDream了解更多。
11 2
|
3天前
|
XML JSON Java
Android面试题 之 网络通信基础面试题
序列化对比:Serializable码流大、性能低;XML人机可读但复杂;JSON轻量、兼容性好但空间消耗大;ProtoBuff高效紧凑。支持大量长连接涉及系统限制调整、缓冲区优化。select/poll/epoll是IO多路复用,epoll在高连接数下性能更优且支持边缘触发。水平触发持续通知数据,边缘触发仅通知新数据。直接内存减少一次拷贝,零拷贝技术如sendfile和MMAP提升效率。关注公众号&quot;AntDream&quot;了解更多技术细节。
7 1
|
3天前
|
Android开发 Kotlin
Android面试题 之 Kotlin DataBinding 图片加载和绑定RecyclerView
本文介绍了如何在Android中使用DataBinding和BindingAdapter。示例展示了如何创建`MyBindingAdapter`,包含一个`setImage`方法来设置ImageView的图片。布局文件使用`&lt;data&gt;`标签定义变量,并通过`app:image`调用BindingAdapter。在Activity中设置变量值传递给Adapter处理。此外,还展示了如何在RecyclerView的Adapter中使用DataBinding,如`MyAdapter`,在子布局`item.xml`中绑定User对象到视图。关注公众号AntDream阅读更多内容。
11 1
|
5天前
|
Android开发
Android面试题之activity启动流程
该文探讨了Android应用启动和Activity管理服务(AMS)的工作原理。从Launcher启动应用开始,涉及Binder机制、AMS回调、进程创建、Application和Activity的生命周期。文中详细阐述了AMS处理流程,包括创建ClassLoader、加载APK、启动Activity的步骤,以及权限校验和启动模式判断。此外,还补充了activity启动流程中AMS的部分细节。欲了解更多内容,可关注公众号“AntDream”。
10 1
|
5天前
|
vr&ar 数据库 Android开发
Android面试题之ActivityManagerService的启动流程
本文探讨了Android系统的SystemServer启动过程,包括创建SystemContext、引导服务、启动各类核心服务以及AMS的启动和初始化。AMS负责管理activity、广播队列、provider等,并设置SystemProcess,安装系统Provider。当AMS调用SystemReady时,系统UI准备启动,启动Launcher。文中还对比了init、zygote和system_server进程的角色。最后推荐了两本关于Android内核剖析的书籍:柯元旦教授的《Android内核剖析》和罗升阳的《Android系统源代码情景分析》。关注公众号AntDream获取更多内容。
7 0
|
6天前
|
JSON 安全 调度
Android面试题之Kotlin协程一文搞定
本文介绍了协程的基础知识,强调它是轻量级线程,用于处理耗时任务而不阻塞主线程,确保主线程安全。协程特点包括使异步逻辑同步化,并允许函数挂起和恢复。挂起函数由`suspend`关键字标识,只能在协程内部调用。挂起与阻塞的主要区别在于挂起不会导致主线程ANR。 结构化并发和协程作用域(如`CoroutineScope`、`GlobalScope`、`MainScope`等)提供了任务管理,文章还探讨了并发、启动模式、协程取消、超时任务以及资源释放等主题。
13 0