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;
}


相关文章
|
4天前
|
传感器 监控 Android开发
构建高效安卓应用:深入理解Android Profiler
【5月更文挑战第28天】在移动开发领域,应用性能优化是一个永恒的话题。随着Android设备的多样化,确保应用在不同设备上都能流畅运行成为了开发者的一项重要任务。Android Profiler是Android Studio提供的一个集成工具,它能够帮助开发者监控应用的CPU、内存、网络和电池使用情况。本文将深入探讨如何使用Android Profiler来分析并优化应用性能,使开发者能够构建出更高效的安卓应用。
10 0
|
15天前
|
消息中间件 安全 关系型数据库
100天!教你逐步突破,快速掌握成为 Python 高手!,安卓面试题及答案
100天!教你逐步突破,快速掌握成为 Python 高手!,安卓面试题及答案
|
15天前
|
XML 前端开发 Android开发
Android架构设计——MVC,滴滴 战略 面试
Android架构设计——MVC,滴滴 战略 面试
|
15天前
|
Android开发 异构计算 前端开发
Android显示原理,安卓自定义view面试
Android显示原理,安卓自定义view面试
|
15天前
|
算法 前端开发 Android开发
Android文字基线Baseline算法的使用讲解,Android开发面试题
Android文字基线Baseline算法的使用讲解,Android开发面试题
Android文字基线Baseline算法的使用讲解,Android开发面试题
|
1天前
|
Android开发 开发者 UED
探索安卓应用开发中的UI设计趋势
随着移动应用市场的不断发展和用户需求的变化,安卓应用的UI设计趋势也在不断演进。本文将深入探讨当前安卓应用开发中的UI设计趋势,包括暗黑模式、原生化设计、动效设计等方面的发展趋势,为开发者提供参考和启发。
|
1天前
|
安全 网络安全 量子技术
网络安全与信息安全:漏洞、加密技术与安全意识的探索安卓应用开发中的内存管理策略
【5月更文挑战第31天】随着互联网的普及,网络安全问题日益严重。本文将深入探讨网络安全漏洞、加密技术以及安全意识等方面的问题,以期提高公众对网络安全的认识和防范能力。
|
2天前
|
机器学习/深度学习 算法 Android开发
安卓应用开发:打造高效通知管理系统
【5月更文挑战第31天】在移动应用的海洋中,用户经常被各种推送通知所困扰。一个精心设计的通知管理系统对于提升用户体验至关重要。本文将探讨如何在安卓平台上开发一个高效的通知管理系统,包括系统设计原则、实现机制以及性能优化策略,旨在为开发者提供一套可行的解决方案,以减少用户干扰并提高应用的用户留存率。
|
2天前
|
缓存 监控 Android开发
Android 开发中的内存优化策略
【5月更文挑战第30天】在移动应用的开发过程中,性能和用户体验始终是核心关注点。对于基于Android平台的应用程序,有效的内存管理是确保流畅运行和优异性能的关键因素之一。本文将深入探讨Android开发中常见的内存问题,并提出一系列实用的内存优化策略。我们将从内存泄漏的识别与防止开始,到合理使用内存缓存技巧,以及高效的数据结构选择等方面进行详细阐述。通过这些策略的实施,开发者可以显著减少应用的内存占用,提升应用的稳定性和响应速度,进而改善最终用户的体验。