2024年Android最新Framework学习(十)Content Provider启动过程(1),字节跳动前端面试问题及答案

总结

笔者之前工作是在金融公司可能并不是特别追求技术,而笔者又是喜欢追求技术的人,所以格格不入,只能把目标放在互联网大厂了。也希望大家都去敢于尝试和追逐自己的梦想!
BATJ大厂Android高频面试题

觉得有收获的记得点赞,关注+收藏哦!你们的点赞就是我的动力!

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

需要这份系统化学习资料的朋友,可以戳这里获取

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

ContentResolver#query()

public final @Nullable Cursor query(final @RequiresPermission.Read @NonNull Uri uri,

@Nullable String[] projection, @Nullable String selection,

@Nullable String[] selectionArgs, @Nullable String sortOrder,

@Nullable CancellationSignal cancellationSignal) {

Preconditions.checkNotNull(uri, “uri”);

IContentProvider unstableProvider = acquireUnstableProvider(uri); //1

try {

try {

qCursor = unstableProvider.query(mPackageName, uri, projection,

selection, selectionArgs, sortOrder, remoteCancellationSignal); //2

} catch (DeadObjectException e) {

}

}

注释1处通过acquireUnstableProvider方法返回IContentProvider类型的unstableProvider对象。

注释2处调用unstableProvider的query方法。

先看看注释1的方法吧。

ContentResolver#acquireUnstableProvider()

public final IContentProvider acquireUnstableProvider(Uri uri) {

if (!SCHEME_CONTENT.equals(uri.getScheme())) { //1

return null;

}

String auth = uri.getAuthority();

if (auth != null) {

return acquireUnstableProvider(mContext, uri.getAuthority()); //2

}

return null;

}

注释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) {

最后

我这里整理了一份完整的学习思维以及Android开发知识大全PDF。

当然实践出真知,即使有了学习线路也要注重实践,学习过的内容只有结合实操才算是真正的掌握。

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

需要这份系统化学习资料的朋友,可以戳这里获取

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

ntentProviders(

getApplicationThread(), results); //3

} catch (RemoteException ex) {

最后

我这里整理了一份完整的学习思维以及Android开发知识大全PDF。

[外链图片转存中…(img-LKTs6UzR-1715666675086)]

当然实践出真知,即使有了学习线路也要注重实践,学习过的内容只有结合实操才算是真正的掌握。

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

需要这份系统化学习资料的朋友,可以戳这里获取

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值