android工程师面试题目和答案,浅析Android插件化,进阶加薪全靠它

}
}

我们在看一下ActivityManagerProxy这个代理。

@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
if (method.getName().equals(“startService”)) {
Intent intent = (Intent) args[1];
List infos = mPackageManager.queryIntentServices(intent, PackageManager.MATCH_ALL);
if (infos == null || infos.size() == 0) {
intent.putExtra(TARGET_SERVICE, intent.getComponent().getClassName());
intent.setClassName(“com.guolei.plugindemo”, “com.guolei.plugindemo.StubService”);
}

}
return method.invoke(mOrigin, args);
}

代码很清晰、也很简单,不需要在做多余的了,那么,我们看下代理Service是如何启动并且调用我们的插件Service的。

@Override
public int onStartCommand(Intent intent, int flags, int startId) {
Log.e(TAG, "onStartCommand: stub service ");
if (intent != null && !TextUtils.isEmpty(intent.getStringExtra(TARGET_SERVICE))) {
//启动真正的service
String serviceName = intent.getStringExtra(TARGET_SERVICE);
try {
Class activityThreadClz = Class.forName(“android.app.ActivityThread”);
Method getActivityThreadMethod = activityThreadClz.getDeclaredMethod(“getApplicationThread”);
getActivityThreadMethod.setAccessible(true);
//获取ActivityThread
Class contextImplClz = Class.forName(“android.app.ContextImpl”);
Field mMainThread = contextImplClz.getDeclaredField(“mMainThread”);
mMainThread.setAccessible(true);
Object activityThread = mMainThread.get(getBaseContext());
Object applicationThread = getActivityThreadMethod.invoke(activityThread);
//获取token值
Class iInterfaceClz = Class.forName(“android.os.IInterface”);
Method asBinderMethod = iInterfaceClz.getDeclaredMethod(“asBinder”);
asBinderMethod.setAccessible(true);
Object token = asBinderMethod.invoke(applicationThread);
//Service的attach方法
Class serviceClz = Class.forName(“android.app.Service”);
Method attachMethod = serviceClz.getDeclaredMethod(“attach”,
Context.class, activityThreadClz, String.class, IBinder.class, Application.class, Object.class);
attachMethod.setAccessible(true);
Class activityManagerNative = Class.forName(“android.app.ActivityManagerNative”);
Field gDefaultField = activityManagerNative.getDeclaredField(“gDefault”);
gDefaultField.setAccessible(true);
Object origin = gDefaultField.get(null);
Class singleton = Class.forName(“android.util.Singleton”);
Field mInstanceField = singleton.getDeclaredField(“mInstance”);
mInstanceField.setAccessible(true);
Object originAMN = mInstanceField.get(origin);
Service targetService = (Service) Class.forName(serviceName).newInstance();
attachMethod.invoke(targetService, this, activityThread, intent.getComponent().getClassName(), token,
getApplication(), originAMN);
//service的oncreate方法
Method onCreateMethod = serviceClz.getDeclaredMethod(“onCreate”);
onCreateMethod.setAccessible(true);
onCreateMethod.invoke(targetService);
targetService.onStartCommand(intent, flags, startId);
} catch (Exception e) {
e.printStackTrace();
Log.e(TAG, "onStartCommand: " + e.getMessage());
}
}
return super.onStartCommand(intent, flags, startId);
}

代码较长,逻辑如下:

  • 检测到需要启动插件Service
  • 构建插件Service attach方法需要的参数
  • 构造一个插件Service
  • 调用插件Service的attach方法
  • 调用插件Service的onCreate方法

这样,一个插件Service就启动起来了。

BroadcastReceiver的插件化

BroadcastReceiver分为两种,静态注册,和动态注册。静态注册的是PMS在安装或者系统启动的时候扫描APK,解析配置文件,并存储在PMS端的,这个我们无法干预,并且,我们的插件由于未安装,静态注册的是无法通过系统正常行为装载的。而动态注册的,由于没有检测这一步,因此,也不需要我们干预。我们现在需要解决的问题就是,怎么能装载插件中静态注册的。

我们可以通过解析配置文件,自己调用动态注册的方法去注册这个。

代码这里就不贴了,和下面ContentProvider的一起贴。

ContentProvider的插件化

和其他三个组件不一样的是,ContentProvider是在进程启动入口,也就是ActivityThread中进行安装的。那么我们可以按照这个思路,自己去进行安装的操作。

代码如下。

Field providersField = packageClz.getDeclaredField(“providers”);
providersField.setAccessible(true);
ArrayList providers = (ArrayList) providersField.get(packageObject);

Class providerClz = Class.forName(“android.content.pm.PackageParser$Provider”);
Field providerInfoField = providerClz.getDeclaredField(“info”);
providersField.setAccessible(true);
List providerInfos = new ArrayList<>();
for (int i = 0; i < providers.size(); i++) {
ProviderInfo providerInfo = (ProviderInfo) providerInfoField.get(providers.get(i));
providerInfo.applicationInfo = getApplicationInfo();
providerInfos.add(providerInfo);
}
Class contextImplClz = Class.forName(“android.app.ContextImpl”);
Field mMainThread = contextImplClz.getDeclaredField(“mMainThread”);
mMainThread.setAccessible(true);
Object activityThread = mMainThread.get(this.getBaseContext());
Class activityThreadClz = Class.forName(“android.app.ActivityThread”);
Method installContentProvidersMethod = activityThreadClz.getDeclaredMethod(“installContentProviders”, Context.class, List.class);
installContentProvidersMethod.setAccessible(true);
installContentProvidersMethod.invoke(activityThread, this, providerInfos);

贴一下整体的代码,这里的代码,包括Multidex方法加dex,BroadcastReceiver的插件化以及ContentProvider的插件化。

private void loadClassByHostClassLoader() {
File apkFile = new File(“/sdcard/plugin_1.apk”);
ClassLoader baseClassLoader = this.getClassLoader();
try {
Field pathListField = baseClassLoader.getClass().getSuperclass().getDeclaredField(“pathList”);

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

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

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

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

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

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

最后送福利了,现在关注我可以获取包含源码解析,自定义View,动画实现,架构分享等。
内容难度适中,篇幅精炼,每天只需花上十几分钟阅读即可。
大家可以跟我一起探讨,有flutter—底层开发—性能优化—移动架构—资深UI工程师 —NDK相关专业人员和视频教学资料,还有更多面试题等你来拿

录播视频图.png

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

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

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

IGC大模型、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、付费专栏及课程。

余额充值