Framework学习(十)Content Provider启动过程(1)

注释1处用来检查Uri的scheme是否等于”content”,如果不是则返回null。

注释2处调用了acquireUnstableProvider方法,这是个抽象方法,它的实现在ContentResolver的子类ApplicationContentResolver中。

frameworks/base/core/java/android/app/ContextImpl.java

ApplicationContentResolver#acquireUnstableProvider()

@Override

protected IContentProvider acquireUnstableProvider(Context c, String auth) {

return mMainThread.acquireProvider(c,

ContentProvider.getAuthorityWithoutUserId(auth),

resolveUserIdFromAuthority(auth), false);

}

返回了ActivityThread类型的mMainThread对象的acquireProvider方法。

frameworks/base/core/java/android/app/ActivityThread.java

ActivityThread#acquireProvider()

public final IContentProvider acquireProvider(

Context c, String auth, int userId, boolean stable) {

final IContentProvider provider = acquireExistingProvider(c, auth, userId, stable); //1

if (provider != null) {

return provider;

}

IActivityManager.ContentProviderHolder holder = null;

try {

holder = ActivityManagerNative.getDefault().getContentProvider(

getApplicationThread(), auth, userId, stable); //2

} catch (RemoteException ex) {

throw ex.rethrowFromSystemServer();

}

if (holder == null) {

Slog.e(TAG, "Failed to find provider info for " + auth);

return null;

}

holder = installProvider(c, holder, holder.info,

true /noisy/, holder.noReleaseNeeded, stable); //3

return holder.provider;

}

注释1处检查ActivityThread中的ArrayMap类型的mProviderMap中是否有目标ContentProvider存在,有则返回,没有就会在注释2处调用AMP的getContentProvider方法,最终会调用AMS的getContentProvider方法。

注释3处的installProvider方法用来将注释2处返回的ContentProvider相关的数据存储在mProviderMap中,起到缓存的作用,这样使用相同的Content Provider时,就不需要每次都要调用AMS的getContentProvider方法。

AMS到ActivityThread的调用


frameworks/base/services/core/java/com/android/server/am/ActivityManagerService.java

ActivityManagerService#getContentProvider()

@Override

public final ContentProviderHolder getContentProvider(

IApplicationThread caller, String name, int userId, boolean stable) {

return getContentProviderImpl(caller, name, null, stable, userId);

}

ActivityManagerService#getContentProviderImpl()

private ContentProviderHolder getContentProviderImpl(IApplicationThread caller,

String name, IBinder token, boolean stable, int userId) {

ProcessRecord proc = getProcessRecordLocked(

cpi.processName, cpr.appInfo.uid, false); //1

if (proc != null && proc.thread != null && !proc.killed) {

if (!proc.pubProviders.containsKey(cpi.name)) {

checkTime(startTime, “getContentProviderImpl: scheduling install”);

proc.pubProviders.put(cpi.name, cpr);

try {

proc.thread.scheduleInstallProvider(cpi); //2

} catch (RemoteException e) {

}

}

} else {

checkTime(startTime, “getContentProviderImpl: before start process”);

proc = startProcessLocked(cpi.processName,

cpr.appInfo, false, 0, “content provider”,

new ComponentName(cpi.applicationInfo.packageName,

cpi.name), false, false, false); //3

checkTime(startTime, “getContentProviderImpl: after start process”);

}

}

注释1处通过getProcessRecordLocked方法来获取目标ContentProvider的应用程序进程信息,这些信息用ProcessRecord类型的proc来表示,如果该应用进程已经启动就会调用注释2处的代码,否则就会调用注释3的startProcessLocked方法来启动进程。

应用程序进程启动过程请参考Framework学习(六)应用程序进程启动过程这篇文章。

ActivityThread启动Provider


frameworks/base/services/core/java/com/android/app/ActivityThread.java

ActivityThread#scheduleInstallProvider()

@Override

public void scheduleInstallProvider(ProviderInfo provider) {

sendMessage(H.INSTALL_PROVIDER, provider);

}

这里的H是ActivityThread的内部类并继承Handler。

ActivityThread.H


private class H extends Handler {

public static final int INSTALL_PROVIDER = 145;

public void handleMessage(Message msg) {

if (DEBUG_MESSAGES) Slog.v(TAG, ">>> handling: " + codeToString(msg.what));

switch (msg.what) {

case INSTALL_PROVIDER:

handleInstallProvider((ProviderInfo) msg.obj);

break;

}

ActivityThread#handleInstallProvider()

public void handleInstallProvider(ProviderInfo info) {

final StrictMode.ThreadPolicy oldPolicy = StrictMode.allowThreadDiskWrites();

try {

installContentProviders(mInitialApplication, Lists.newArrayList(info)); //1

} finally {

StrictMode.setThreadPolicy(oldPolicy);

}

}

注释1调用了installContentProviders方法。

ActivityThread#installContentProviders()

private void installContentProviders(

Context context, List providers) {

final ArrayList<IActivityManager.ContentProviderHolder> results =

new ArrayList<IActivityManager.ContentProviderHolder>();

for (ProviderInfo cpi : providers) { //1

if (DEBUG_PROVIDER) {

StringBuilder buf = new StringBuilder(128);

buf.append("Pub ");

buf.append(cpi.authority);

buf.append(": ");

buf.append(cpi.name);

Log.i(TAG, buf.toString());

}

IActivityManager.ContentProviderHolder cph = installProvider(context, null, cpi,

false /noisy/, true /noReleaseNeeded/, true /stable/); //2

if (cph != null) {

cph.noReleaseNeeded = true;

results.add(cph);

}

}

try {

ActivityManagerNative.getDefault().publishContentProviders(

getApplicationThread(), results); //3

} catch (RemoteException ex) {

throw ex.rethrowFromSystemServer();

}

}

注释1处遍历当前应用程序进程的ProviderInfo列表,得到每个Content Provider的ProviderInfo(存储Content Provider的信息)。

注释2处调用installProvider方法来启动这些Content Provider。

注释3处通过AMS的publishContentProviders方法将这些Content Provider存储在AMS的mProviderMap中,这个mProviderMap在前面提到过,起到缓存的作用,防止每次使用相同的Content Provider时都会调用AMS的getContentProvider方法。

ActivityThread#installProvider()

private IActivityManager.ContentProviderHolder installProvider(Context context,

IActivityManager.ContentProviderHolder holder, ProviderInfo info,

boolean noisy, boolean noReleaseNeeded, boolean stable) {

ContentProvider localProvider = null;

final java.lang.ClassLoader cl = c.getClassLoader();

localProvider = (ContentProvider)cl.

loadClass(info.name).newInstance(); //1

provider = localProvider.getIContentProvider();

if (provider == null) {

return null;

}

if (DEBUG_PROVIDER) Slog.v(

TAG, "Instantiating local provider " + info.name);

localProvider.attachInfo(c, info); //2

} catch (java.lang.Exception e) {

}

return null;

}

}

return retHolder;

}

注释1处通过反射来创建ContentProvider类型的localProvider对象。

注释2处调用了它的attachInfo方法。

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

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

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

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

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

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

建议

当我们出去找工作,或者准备找工作的时候,我们一定要想,我面试的目标是什么,我自己的技术栈有哪些,近期能掌握的有哪些,我的哪些短板 ,列出来,有计划的去完成,别看前两天掘金一些大佬在驳来驳去 ,他们的观点是他们的,不要因为他们的观点,膨胀了自己,影响自己的学习节奏。基础很大程度决定你自己技术层次的厚度,你再熟练框架也好,也会比你便宜的,性价比高的替代,很现实的问题但也要有危机意识,当我们年级大了,有哪些亮点,与比我们经历更旺盛的年轻小工程师,竞争。

  • 无论你现在水平怎么样一定要 持续学习 没有鸡汤,别人看起来的毫不费力,其实费了很大力,这四个字就是我的建议!!!!!!!!!

  • 准备想说怎么样写简历,想象算了,我觉得,技术就是你最好的简历

  • 我希望每一个努力生活的it工程师,都会得到自己想要的,因为我们很辛苦,我们应得的。

  • 有什么问题想交流,欢迎给我私信,欢迎评论

【附】相关架构及资料

Android高级技术大纲

面试资料整理

内含往期Android高级架构资料、源码、笔记、视频。高级UI、性能优化、架构师课程、NDK、混合式开发(ReactNative+Weex)微信小程序、Flutter全方面的Android进阶实践技术

,也会比你便宜的,性价比高的替代,很现实的问题但也要有危机意识,当我们年级大了,有哪些亮点,与比我们经历更旺盛的年轻小工程师,竞争。

  • 无论你现在水平怎么样一定要 持续学习 没有鸡汤,别人看起来的毫不费力,其实费了很大力,这四个字就是我的建议!!!!!!!!!

  • 准备想说怎么样写简历,想象算了,我觉得,技术就是你最好的简历

  • 我希望每一个努力生活的it工程师,都会得到自己想要的,因为我们很辛苦,我们应得的。

  • 有什么问题想交流,欢迎给我私信,欢迎评论

【附】相关架构及资料

[外链图片转存中…(img-9bebHiH2-1711935919459)]

[外链图片转存中…(img-dvdp9RCp-1711935919459)]

内含往期Android高级架构资料、源码、笔记、视频。高级UI、性能优化、架构师课程、NDK、混合式开发(ReactNative+Weex)微信小程序、Flutter全方面的Android进阶实践技术

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

  • 18
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值