Android Framework学习笔记(十)Content Provider启动过程

先自我介绍一下,小编浙江大学毕业,去过华为、字节跳动等大厂,目前阿里P7

深知大多数程序员,想要提升技能,往往是自己摸索成长,但自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

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

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

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

如果你需要这些资料,可以添加V获取:vip204888 (备注Android)
img

正文

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方法。

frameworks/base/core/java/android/content/ContentProvider.java

ContentProvider#attachInfo()


最后

这里我特地整理了一份《Android开发核心知识点笔记》,里面就包含了自定义View相关的内容

除了这份笔记,还给大家分享 Android学习PDF+架构视频+面试文档+源码笔记,高级架构技术进阶脑图、Android开发面试专题资料,高级进阶架构资料这几块的内容。非常适合近期有面试和想在技术道路上继续精进的朋友。

分享上面这些资源,希望可以帮助到大家提升进阶,如果你觉得还算有用的话,不妨把它们推荐给你的朋友~

喜欢本文的话,给我点个小赞、评论区留言或者转发支持一下呗~

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化的资料的朋友,可以添加V获取:vip204888 (备注Android)
img

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

了这份笔记,还给大家分享 Android学习PDF+架构视频+面试文档+源码笔记,高级架构技术进阶脑图、Android开发面试专题资料,高级进阶架构资料这几块的内容。非常适合近期有面试和想在技术道路上继续精进的朋友。

[外链图片转存中…(img-wDZYRzEJ-1713437537034)]

分享上面这些资源,希望可以帮助到大家提升进阶,如果你觉得还算有用的话,不妨把它们推荐给你的朋友~

喜欢本文的话,给我点个小赞、评论区留言或者转发支持一下呗~

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化的资料的朋友,可以添加V获取:vip204888 (备注Android)
[外链图片转存中…(img-FC8fU9td-1713437537034)]

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值