(转)使用 ContentProvider 无侵入获取 Context


转://www.greatytc.com/p/8f7ba9e88cd0文章

前言

在Android中,使用三方库或二方库时,通常都需要使用Context进行初始化

这篇文章将介绍一种基于ContentProvider机制实现的无侵入获取Context的方法,希望能帮上忙

目录

1. 获取 Context 的常规方法

我们知道Context本身是一个抽象类,所以我们获取Context实际上是获取Context的实现类,具体可以分为:Application、Activity、Service与ContextImpl。若还不了解,请务必阅读文章:《Android | 一个进程有多少个 Context 对象(答对的不多)》

1.1 获取 Application 对象

既然Application是Context的实现类,那么我们就可以直接使用Application对象来初始化第三方库,同时也可以使用一个静态方法将对象暴露出去:

// MainApplication.ktclassMainApplication:Application(){companionobject{lateinitvarapplication:Applicationget}overridefunonCreate(){super.onCreate()application=this// 初始化第三方库}}

1.2 获取 Activity & Service 对象

同样地,Activity & Service也是Context的实现类,那么我们就可以在程序运行过程中,按需初始化第三方库。例如使用Glide时,并不需要一开始就调用Glide#with(Context),只需要在显示图片的时候调用即可:

【todo】

1.3 小结

优点

最常用的方式,实现简单,没有性能 / 稳定性风险;可以按需初始化第三方库 & 懒加载

缺点

需要获取ApplicationContext / Context(依赖方与库代码强耦合),不利于组件化

下面,我将介绍两种无侵入获取Context的方法,将涉及到Android进程的启动流程,若还不了解,请务必阅读文章:《Android | 带你理解 Application 的创建过程》

2. 反射 ActivityThread 获得 ApplicationContext(不推荐)

这一节介绍一种通过ActivityThread.java获得Application的方法,具体如下:

2.1 源码分析

我们都知道,在启动四大组件(Activity、Service、ContentProvider, BroadcastReceiver)时,如果对应的进程未启动,就需要先创建进程,相应地也会创建一个Application对象。若还不了解,请务必阅读:《Android | 带你理解 Application 的创建过程》。简单来说:

在system_server进程,通过AMS#getProcessRecordLocked(...)获取进程信息(ProcessRecord);

若不存在,则调用AMS#startProcessLocked(...)创建进程

在Zygote孵化目标进程之后,在目标进程反射执行ActivityThread#main(),并最终在ActivityThread#handleBindApplication(...)中创建Application对象

// ActivityThread.javaApplication mInitialApplication;publicApplicationgetApplication(){returnmInitialApplication;}privatevoidhandleBindApplication(AppBindDatadata){// ...Application app;// data.info 为 LoadedApk.javaapp=data.info.makeApplication(data.restrictedBackupMode,null);// ...mInitialApplication=app;// ...}

可以看到,创建Application对象之后会保存在mInitialApplication属性中,那么如果我们可以访问到这个属性,是不是就可以获得Application对象了呢?

首先,我们需要获得ActivityThread对象,那么我们先在源码中寻找创建ActivityThread对象的地方:

// ActivityThread.javaprivatestaticvolatileActivityThreadsCurrentActivityThread;publicstaticActivityThreadcurrentActivityThread(){returnsCurrentActivityThread;}// (简化)publicstaticvoidmain(String[]args){Looper.prepareMainLooper();// 创建 ActivityThread 对象ActivityThreadthread=newActivityThread();thread.attach(false,startSeq);Looper.loop();}privatevoidattach(booleansystem,longstartSeq){sCurrentActivityThread=this;}

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

2.2 使用步骤

// 新建文件 Context.ktprivatevarapplication:Context?=nullfuncontext():Context{if(null==application){try{valactivityThread:Anyvalclazz=Class.forName("android.app.ActivityThread")valcurrentActivityThread=clazz.getMethod("currentActivityThread").apply{isAccessible=true}valgetApplication=clazz.getMethod("getApplication").apply{isAccessible=true}activityThread=currentActivityThread.invoke(null)application=getApplication.invoke(activityThread)asContext}catch(e:Throwable){// 存在未适配的风险}}returnapplication!!}

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

android.app.Application@c12661f

2.3 小结

优点:

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

缺点:

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

3. 使用 ContentProvider 获取 ApplicationContext

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

3.1 源码分析

// ActivityThread.javaprivatevoidhandleBindApplication(AppBindDatadata){// ...Applicationapp;app=data.info.makeApplication(data.restrictedBackupMode,null);mInitialApplication=app;// 初始化所有 ContentProviderinstallContentProviders(app,data.providers);// ...}privatevoidinstallContentProviders(Contextcontext,List<ProviderInfo>providers){final ArrayList<ContentProviderHolder>results=newArrayList<>();for(ProviderInfocpi:providers){// 依次初始化 ContentProviderContentProviderHoldercph=installProvider(context,null,cpi,false/*noisy*/,true/*noReleaseNeeded*/,true/*stable*/);if(cph!=null){cph.noReleaseNeeded=true;results.add(cph);}}// ...}

可以看到,在ActivityThread中,创建Application对象之后会调用installContentProviders()安装属于当前进程(processName)的ContentProvider,若还不了解,请务必阅读文章:《Android | 带你理解 ContentProvider 机制》

在ContentProvider声明的方法中,提供了getContext()获得ApplicationContext,所以,我们就可以ContentProvider启动的机制,从ContentProvider启动时拿到ApplicationContext

3.2 使用步骤

步骤一:实现 ContentProvider 子类

// ContextProvider.ktinternalclassContextProvider:ContentProvider(){overridefunonCreate():Boolean{init(context!!)returntrue}// 其他方法直接 return }// Context.ktprivatelateinitvarapplication:Contextfuninit(context:Context){application=context}funcontext():Context{returnapplication}

步骤二:在 AndroidManifest 中配置

// AndroidManifest.xml<application>    <providerandroid:name=".Contextprovider"android:authorities="${applicationId}.contextprovider"android:exported="false"/></application>

步骤三:使用

Toast.makeText(context(),"",Toast.LENGTH_SHORT).show()

3.3 小结

优点:

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

缺点:

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

风险:

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

4. 案例

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

LeakCanary 2.4

internalsealedclassAppWatcherInstaller:ContentProvider(){internalclassMainProcess:AppWatcherInstaller()internalclassLeakCanaryProcess:AppWatcherInstaller()overridefunonCreate():Boolean{valapplication=context!!.applicationContextasApplication        AppWatcher.manualInstall(application)returntrue}// 其他方法直接 return }

AutoSize 1.1.2

publicclassInitProviderextendsContentProvider{@OverridepublicbooleanonCreate(){AutoSizeConfig.getInstance().setLog(true).init((Application)getContext().getApplicationContext()).setUseDeviceSize(false);returntrue;}// 其他方法直接 return }

Picasso 2.7

publicfinalclassPicassoProviderextendsContentProvider{@SuppressLint("StaticFieldLeak")staticContextcontext;@OverridepublicbooleanonCreate(){context=getContext();returntrue;}// 其他方法直接 return }

作者:彭旭锐

链接://www.greatytc.com/p/8f7ba9e88cd0

来源:简书

著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 211,948评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,371评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,490评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,521评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,627评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,842评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,997评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,741评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,203评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,534评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,673评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,339评论 4 330
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,955评论 3 313
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,770评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,000评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,394评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,562评论 2 349