尊重原创:http://blog.csdn.net/yuanzeyao/article/details/42243583
在前面的一篇文章中,我们详细分析了PackageManagerService的启动过程(在后面的文章中,为了方便,我会将PackageManagerService简称PMS),PMS在启动的过程中,会去扫描系统app和用户安装的app,然后将这些app的信息保存到一些数据结构中,在这篇文章中,我们会接着前面一篇文章继续分析Intent匹配查询过程,如果对PMS不是很熟悉的同学建议先去阅读前面一篇文章PackageManagerService启动过程分析。
作为一名Android App开发着,我相信你对Intent的使用是再熟悉不过了,例如我想在一个Activity中启动另外一个Activity,会使用如下代码:
Intent intent=new Intent(this,SecondActivity.class);
this.startActivity(intent);
以上方式称为显示Intent调用,当然有些时候我们会使用隐式Intent,例如:
Intent intent=new Intent("com.android.demo");
this.startActivity(intent);
由于Intent的使用非常简单,所以在这里我不想再去花太多时间去描述它了,我们这里是要从源码的角度去理解通过Intent是如何匹配Acitivity的(Service,Receiver原理也是差不多的)。
我们直接从startActivity函数开始吧(提示:我使用的是4.1源码,不同版本的源码会有些出入),在这里,先给出一张时序图,然后跟着时序图看源码。
图1-1
根据图1-1,当我们调用Activity的startActivity方法时,其实调用的就是调用ContextImpl的startActivity方法
public void startActivity(Intent intent, Bundle options) {
if ((intent.getFlags()&Intent.FLAG_ACTIVITY_NEW_TASK) == 0) {
throw new AndroidRuntimeException(
"Calling startActivity() from outside of an Activity "
+ " context requires the FLAG_ACTIVITY_NEW_TASK flag."
+ " Is this really what you want?");
}
mMainThread.getInstrumentation().execStartActivity(
getOuterContext(), mMainThread.getApplicationThread(), null,
(Activity)null, intent, -1, options);
}
在ContextImple的startActivity方法中,会调用Instrumentation的execStartActivity方法,这个方法我就不贴出源码了,它里面其实就是调用了ActivityManagerService的startActivity方法,这个方法里面其实就是调用了ActivityStack的startActivityMayWait方法,该方法又调用了自身的resolveActivity方法,最后调用了PMS的resolveIntent方法了,到这里终于见到了PMS了,在resolveIntent方法里面就是调用了自身的queryIntentActivities方法,queryIntentActivities会返回一个ActivityInfo对象,我们知道一个ActivityInfo对象就是一个Activity的档案对象,记录了一个Acitivity的所有的信息。这里给出queryIntentActivities的源码
public List<ResolveInfo> queryIntentActivities(Intent intent,
String resolvedType, int flags, int userId) {
if (!sUserManager.exists(userId)) return null;
ComponentName comp = intent.getComponent();
if (comp == null) {
if (intent.getSelector() != null) {
intent = intent.getSelector();
comp = intent.getComponent();
}
}
if (comp != null) {
final List<ResolveInfo> list = new ArrayList<ResolveInfo>(1);
final ActivityInfo ai = getActivityInfo(comp, flags, userId);
if (ai != null) {
final ResolveInfo ri = new ResolveInfo();
ri.activityInfo = ai;
list.add(ri);
}
return list;
}
synchronized (mPackages) {
final String pkgName = intent.getPackage();
if (pkgName == null) {
return mActivities.queryIntent(intent, resolvedType, flags, userId);
}