Android---使用-ContentProvider-无侵入获取-Context,【一步教学,一步到位
- 优点
最常用的方式,实现简单,没有性能 / 稳定性风险;可以按需初始化第三方库 & 懒加载
- 缺点
需要获取ApplicationContext / Context(依赖方与库代码强耦合),不利于组件化
下面,我将介绍两种无侵入获取Context的方法,将涉及到Android进程的启动流程
,若还不了解,请务必阅读文章:《Android | 带你理解 Application 的创建过程》
2. 反射 ActivityThread 获得 ApplicationContext(不推荐)
这一节介绍一种通过ActivityThread.java获得Application的方法,具体如下:
2.1 源码分析
我们都知道,在启动四大组件(Activity、Service、ContentProvider, BroadcastReceiver)时,如果对应的进程未启动,就需要先创建进程,相应地也会创建一个Application对象,即:
- 在system_server进程,通过AMS#getProcessRecordLocked(…)获取进程信息(ProcessRecord);
- 若不存在,则调用AMS#startProcessLocked(…)创建进程;
- 在Zygote孵化目标进程之后,在目标进程反射执行ActivityThread#main(),并最终在ActivityThread#handleBindApplication(…)中创建Application对象。
ActivityThread.java
Application mInitialApplication;
public Application getApplication() {
return mInitialApplication;
}
private void handleBindApplication(AppBindData data) {
// …
Application app;
// data.info 为 LoadedApk.java
app = data.info.makeApplication(data.restrictedBackupMode, null);
// …
mInitialApplication = app;
// …
}
复制代码
可以看到,创建Application对象之后会保存在mInitialApplication属性中,那么如果我们可以访问到这个属性,是不是就可以获得Application对象了呢?
首先,我们需要获得ActivityThread对象,那么我们先在源码中寻找创建ActivityThread对象的地方:
ActivityThread.java
private static volatile ActivityThread sCurrentActivityThread;
public static ActivityThread currentActivityThread() {
return sCurrentActivityThread;
}
// (简化)
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
复制代码
步骤三:使用
// AndroidManifest.xml
复制代码
步骤三:使用
Android---使用-ContentProvider-无侵入获取-Context,【一步教学,一步到位相关推荐
- Android---使用-ContentProvider-无侵入获取-Context,flutter安装配置
class MainApplication : Application() { companion object { lateinit var application: Application get ...
- Android免Root无侵入AOP框架Dexposed
Dexposed框架是阿里巴巴无线事业部近期开源的一款在Android平台下的免Root无侵入运行期AOP框架,该框架基于AOP思想,支持经典的AOP使用场景,可应用于日志记录,性能统计,安全控制,事 ...
- Android---使用-ContentProvider-无侵入获取-Context
Application mInitialApplication; public Application getApplication() { return mInitialApplication; } ...
- Android逆向安全-无侵入找关键call之trace日志分析大法
标题 找关键call是逆向的基本技能和分析目标,找到关键call后便可以进一步利用.在安卓App的逆向分析中,人肉逆向分析虽说不难,但是繁琐,特别是现在App体积动辄几十MB甚至几百MB,反编译出的j ...
- android 获取全局context,说说 Android 中如何在全局获取 Context
Android 提供了一个 Application 类,每当应用启动时,系统就会初始化该类.我们可以定制一个基础的 Application 类,以便管理应用内的一些全局信息. 定义 BaseAppli ...
- Android新技术学习——阿里巴巴免Root无侵入AOP框架Dexposed
阿里巴巴无线事业部近期开源的Android平台下的无侵入运行期AOP框架Dexposed,该框架基于AOP思想,支持经典的AOP使用场景.可应用于日志记录,性能统计,安全控制.事务处理.异常处理等方面 ...
- android获取context的方法,Android编程获取全局Context的方法
Android编程获取全局Context的方法 本文实例讲述了Android编程获取全局Context的方法.分享给大家供大家参考,具体如下: 有时,在处理业务逻辑的时候,需要Context对象,但在 ...
- Dexposed:Android平台免Root无侵入AOP框架
本文来自阿里巴巴技术协会(ATA) 本文首发于 http://www.infoq.com/cn/news/2015/07/dexposed 近日,阿里巴巴无线事业部推出首个重量级Android开源项目 ...
- Android中怎样在工具类中获取Context对象
场景 Android程序中访问资源时需要提供Context,一般来说只有在各种component中(Activity, Provider等等)才能方便的使用api来获取Context对象, 如果在编写 ...
最新文章
- 基于Vue开发的购物车案例
- 《C语言及程序设计》实践项目——画分支结构流程图
- 嵌入式CGI开发之旅——CGI环境变量
- 3dmax挤出制作窗花_给想学3dmax,又不知如何快速入手的你 新手学习3dmax的建议...
- 构造函数与折构函数(c++细节篇五)
- Ubuntu Server 16.04 LTS上使用Docker部署Tomcat
- go使用for...range遍历数组
- python把汉字转换为二进制数_在Python中,如何将8位二进制数转换为ASCII字符?
- 鸿蒙系统下拉菜单,鸿蒙的js开发部模式17:鸿蒙的系统能力的应用模块
- stripe pay_J2Pay –完整示例
- 解决从其它搜索引擎不能直接访问百度页面的问题
- 简单粗暴地开个十层循环(洛谷P2089题题解,Java语言描述)
- 组图:2007最震撼人心的“史上最牛”事件
- arp协议属于哪一层_网络工程师(3):详解ARP协议
- dot全称_游戏dot是什么
- LInux下如何挂载光盘找rpm包?
- CISCO寄存器配置与说明
- [转]linux 系统 errno.h错误码
- 【Linux】目录中 / 和 ~ 的区别
- C++ UDP socket编程