Android---使用-ContentProvider-无侵入获取-Context

文章介绍了如何通过反射获取ActivityThread中的ApplicationContext,以及通过ContentProvider在应用启动时获取ApplicationContext,以减少代码耦合。同时讨论了这种方法的优点、缺点和潜在风险。结尾部分提到了这些技术在面试中的重要性。
摘要由CSDN通过智能技术生成

// (简化)
public static void main(String[] args) {
Looper.prepareMainLooper();

// 创建 ActivityThread 对象
ActivityThread thread = new ActivityThread();
thread.attach(false, startSeq);

Looper.loop();
}

private void attach(boolean system, long startSeq) {
sCurrentActivityThread = this;
}
复制代码

可以看到,ActivityThread对象存储在静态变量sCurrentActivityThread中,那么我们就可以写反射代码了。

2.2 使用步骤

// 新建文件 Context.kt

private var application: Context? = null

fun context(): Context {
if (null == application) {
try {
val activityThread: Any
val clazz = Class.forName(“android.app.ActivityThread”)
val currentActivityThread = clazz.getMethod(“currentActivityThread”).apply {
isAccessible = true
}
val getApplication = clazz.getMethod(“getApplication”).apply {
isAccessible = true
}
activityThread = currentActivityThread.invoke(null)
application= getApplication.invoke(activityThread) as Context
} catch (e: Throwable) {
// 存在未适配的风险
}
}
return application!!
}
复制代码

运行测试一下,context()的返回结果:

android.app.Application@c12661f
复制代码

2.3 小结

  • 优点:

依赖方不需要传递Context对象给库进行初始化,减少了代码耦合,有利于组件化

  • 缺点:

需要反射调用私有API,存在系统版本适配风险;使用反射有一定性能损耗


3. 使用 ContentProvider 获取 ApplicationContext

这一节介绍一种通过ContentProvider.java获得Application的方法。ContentProvider通常的用法是为当前进程 / 远程进程提供内容服务,它们会在应用启动的时候初始化,正因如此,我们可以利用ContentProvider来获得Context。

3.1 源码分析

ActivityThread.java

private void handleBindApplication(AppBindData data) {
// …
Application app;
app = data.info.makeApplication(data.restrictedBackupMode, null);
mInitialApplication = app;
// 初始化所有 ContentProvider
installContentProviders(app, data.providers);
// …
}

private void installContentProviders(Context context, List providers) {
final ArrayList results = new ArrayList<>();

for (ProviderInfo cpi : providers) {
// 依次初始化 ContentProvider
ContentProviderHolder cph = installProvider(context, null, cpi,
false /noisy/, true /noReleaseNeeded/, true /stable/);
if (cph != null) {
cph.noReleaseNeeded = true;
results.add(cph);
}
}
// …
}
复制代码

可以看到,在ActivityThread中,创建Application对象之后会调用installContentProviders()安装属于当前进程(processName)的ContentProvider;而在ContentProvider声明的方法中,提供了getContext()获得ApplicationContext,所以,我们就可以ContentProvider启动的机制,从ContentProvider启动时拿到ApplicationContext

3.2 使用步骤

步骤一:实现 ContentProvider 子类

// ContextProvider.kt

internal class ContextProvider : ContentProvider(){

override fun onCreate(): Boolean {
init(context!!)
return true
}

// 其他方法直接 return
}
// Context.kt
private lateinit var application : Context

fun init(context : Context){
application= context
}

fun context() : Context{
return application
}
复制代码

步骤二:在 AndroidManifest 中配置

// AndroidManifest.xml

复制代码

步骤三:使用

Toast.makeText(context(),“”,Toast.LENGTH_SHORT).show()
复制代码

3.3 小结

  • 优点:

依赖方不需要传递Context对象给库进行初始化,减少了代码耦合,有利于组件化

  • 缺点:

在App 启动时就初始化ContentProvider,不是懒初始化

  • 风险:

应保证初始化非常轻量,否则会降低App的启动速度


4. 案例

下面举出一些基于ContentProvider机制实现无侵入地获取Context的例子:

  • LeakCanary 2.4

AppWatcherInstaller.java

internal sealed class AppWatcherInstaller : ContentProvider() {

internal class MainProcess : AppWatcherInstaller()

internal class LeakCanaryProcess : AppWatcherInstaller()

override fun onCreate(): Boolean {
val application = context!!.applicationContext as Application
AppWatcher.manualInstall(application)
return true
}

// 其他方法直接 return
}
复制代码

  • AutoSize 1.1.2

InitProvider.java

public class InitProvider extends ContentProvider {
@Override
public boolean onCreate() {
AutoSizeConfig.getInstance()
.setLog(true)
.init((Application) getContext().getApplicationContext())
.setUseDeviceSize(false);
return true;
}

// 其他方法直接 return
}
复制代码

  • Picasso 2.7

PicassoProvider.java

public final class PicassoProvider extends ContentProvider {

@SuppressLint(“StaticFieldLeak”) static Context context;

@Override public boolean onCreate() {
context = getContext();
return true;
}

// 其他方法直接 return
}

写在最后

今天关于面试的分享就到这里,还是那句话,有些东西你不仅要懂,而且要能够很好地表达出来,能够让面试官认可你的理解,例如Handler机制,这个是面试必问之题。有些晦涩的点,或许它只活在面试当中,实际工作当中你压根不会用到它,但是你要知道它是什么东西。

最后在这里小编分享一份自己收录整理上述技术体系图相关的几十套腾讯、头条、阿里、美团等公司的面试题,把技术点整理成了视频和PDF(实际上比预期多花了不少精力),包含知识脉络 + 诸多细节,由于篇幅有限,这里以图片的形式给大家展示一部分。

还有 高级架构技术进阶脑图、Android开发面试专题资料,高级进阶架构资料 帮助大家学习提升进阶,也节省大家在网上搜索资料的时间来学习,也可以分享给身边好友一起学习。

【Android核心高级技术PDF文档,BAT大厂面试真题解析】

【算法合集】

【延伸Android必备知识点】

《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》点击传送门,即可获取!
中…(img-jkuYbjH1-1714458254750)]

【算法合集】

[外链图片转存中…(img-Vg4SsOHV-1714458254751)]

【延伸Android必备知识点】

[外链图片转存中…(img-gj0dQFeZ-1714458254752)]

《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》点击传送门,即可获取!

  • 28
    点赞
  • 20
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值