【啃源码】PackageManager-及其应用范例LauncherActivity

PackageManager

官方文档是如下描述PackageManager的

Class for retrieving various kinds of information related to the application packages that are currently installed on the device. You can find this class through Context#getPackageManager.

翻译下:用于检索与设备上当前安装的应用程序包相关的各种信息的类
还是一脸懵逼?不要怀疑自己的理解能力有问题,怪我英语水平太蹩脚。

要理解这句话,重要的一点就是“各种信息”指的是什么。这就要说到我们重要的AndroidManifest.XML文件,这两个东西可以说是息息相关的,我们知道:Application、Activity、Service、Provider都是注册在AndroidManifest里面的,而这些信息就是我们这个PackageManager所管理的信息(包含但不仅限于,比如版本号等)。
简而言之,通过PackageManager你可以拿到注册在AndroidManifest文件中的各种信息:应用图标、包名、所有的Activity、Service信息等等

本篇我们先只分析PackageManager中与Intent相关的的检索方法。
打开PackageManager文件我们可以看到他提供了一系列query开头的方法:

image.png

(哦,忘了说明,本篇代码出自api28,不同版本会有出入。)
以上方法都会根据传入的筛选条件返回一个检索结果集合,我们暂时只关注下返回值是ResolveInfo的集合的这些。这个ResolveInfo是啥玩意?

ResolveInfo

ResolveInfo的基本结构如下

public class ResolveInfo implements Parcelable {
    public ActivityInfo activityInfo;
    public ServiceInfo serviceInfo;
    public ProviderInfo providerInfo;
    public AuxiliaryResolveInfo auxiliaryInfo;
    public boolean isInstantAppAvailable;
...... //后面还有很多,感兴趣的可以自己查看
}

看到类里面的东西就能知道ResolveInfo是对以上提到的注册在AndroidManifest中信息的封装类。

query...()方法

想要知道ResolveInfo从哪里来我们就要看下query...()方法的实现,PackageManager的实现类是DefaultPackageManager,在这个类里我们可以看到是维护了

  private final Map<Intent, List<ResolveInfo>> resolveInfoForIntent = new TreeMap<>(new IntentComparator());

一个以Intent为Key的Map,这也就是为什么这些方法都需要一个Intent对象参数的原因。
当我们在AndroidManifest里面注册一个“信息”时,就会以其所对应的Intent为key加入到resolveInfoForIntent里面,所以我们就可以通过筛选Intent来检索我们需要的ResolveInfo。

PS:我们在AndroidManifest里注册一个Activity的时候会有属性放到<intent-filter>标签里面的,“intent-filter”这个名字就是对应这里筛选的意义,这也确实就是这个标签的意义,<intent-filter>里声明的key-values就是Intent筛选条件

Intent作为key,那我们肯定要看下Intent的equals方法看下怎样才是相同的Key:

//Intent.java

@Override
public boolean equals(Object obj) {
            if (obj instanceof FilterComparison) {
                Intent other = ((FilterComparison)obj).mIntent;
                return mIntent.filterEquals(other);
            }
            return false;
        }
        
        
public boolean filterEquals(Intent other) {
        if (other == null) {
            return false;
        }
        if (!Objects.equals(this.mAction, other.mAction)) return false;
        if (!Objects.equals(this.mData, other.mData)) return false;
        if (!Objects.equals(this.mType, other.mType)) return false;
        if (!Objects.equals(this.mPackage, other.mPackage)) return false;
        if (!Objects.equals(this.mComponent, other.mComponent)) return false;
        if (!Objects.equals(this.mCategories, other.mCategories)) return false;

        return true;
    }

现在你知道为什么声明

<intent-filter>
   <action android:name="android.intent.action.MAIN" />
   <category android:name="android.intent.category.LAUNCHER" />
</intent-filter>

之后就可以作为应用第一个打开的页面了吧,在启动一个app时,就是通过PackageManager检索出符合

action = android.intent.action.MAIN
category = android.intent.category.LAUNCHER

的ResolveInfo,再加以处理。要了解这一过程,我们就要来看sdk提供给我们的LauncherActivity了。

LauncherActivity

Displays a list of all activities which can be performed for a given intent. Launches when clicked.
显示可以针对给定意图执行的所有活动的列表。单击时启动。

LauncherActivity继承自ListActivity,通过ListView实现的一个Intent列表,已经封装了ui。
直接看代码:
当点击某个应用时调用了onListItemClick方法:

    protected void onListItemClick(ListView l, View v, int position, long id) {
        Intent intent = intentForPosition(position);
        startActivity(intent);
    }

intentForPosition方法返回的是一个熟悉的Intent对象,调用了熟悉的startActivity方法去启动这个Intent。

Intent是从ActivityAdapter的intentForPosition方法拿到的:

protected Intent intentForPosition(int position) {
    ActivityAdapter adapter = (ActivityAdapter) mAdapter;
    return adapter.intentForPosition(position); //调用了下面ActivityAdapter中的方法
}

为什么可以拿到一个Intent对象呢?我们看ActivityAdapter这个列表到底维护的数据是什么

 private class ActivityAdapter extends BaseAdapter implements Filterable {
  
        protected List<ListItem> mActivitiesList;
        
        ......
        
        public ActivityAdapter(IconResizer resizer) {
             ......
            mActivitiesList = makeListItems();
        }
        
       ......
       
       //从这里得到的Intent
        public Intent intentForPosition(int position) {
            if (mActivitiesList == null) {
                return null;
            }
            //通过从ListItem取出属性创建出的Intent
            Intent intent = new Intent(mIntent);
            ListItem item = mActivitiesList.get(position);
            intent.setClassName(item.packageName, item.className);
            if (item.extras != null) {
                intent.putExtras(item.extras);
            }
            return intent;
        }
}

可以看到ActivityAdapter维护的数据是一个ListItem的List,最后通过从ListItem取出属性创建出的Intent,这是什么东西?我们先看这玩意从哪里取出来的,上面精简的关键代码中可以看到是通过makeListItems()拿到的:

    public List<ListItem> makeListItems() {
        // Load all matching activities and sort correctly
        //通过mIntent取出List<ResolveInfo>
        List<ResolveInfo> list = onQueryPackageManager(mIntent);
        //排序
        onSortResultList(list);
        //下面逻辑是将ResolveInfo转成ListItem集合
        ArrayList<ListItem> result = new ArrayList<ListItem>(list.size());
        int listSize = list.size();
        for (int i = 0; i < listSize; i++) {
            ResolveInfo resolveInfo = list.get(i);
            //ResolveInfo最后被塞到ListItem里面去了
            result.add(new ListItem(mPackageManager, resolveInfo, null));
        }

        return result;
    }
    
    protected List<ResolveInfo> onQueryPackageManager(Intent queryIntent) {
        return mPackageManager.queryIntentActivities(queryIntent, /* no flags */ 0);
    }

PackageManager调用queryIntentActivities()取到符合mIntent的检索条件的ResolveInfo集合,再将每个ResolveInfo塞到一个新创建的ListItem对象,ListItem是对ResolveInfo的二次封装。

到此,整个流程已经明朗了:

1.通过PackageManager检索以mIntent为筛选条件的ResolveInfo集合
2.对ResolveInfo集合二次封装成ListItem集合,成为列表Adapter的数据
3.当点击条目时,通过ListItem创建出跳转Intent,调用startActivity跳转

回顾一下:

       Intent intent = new Intent(mIntent);
       ListItem item = mActivitiesList.get(position);
       intent.setClassName(item.packageName, item.className);

扩展

LauncherActivity中的mIntent是通过getTargetIntent()方法创建出来的。所以我们可以创建一个Activity继承自LauncherActivity然后重写getTargetIntent方法,将这个条件Intent的Category设置成"android.intent.category.LAUNCHER":

class MainActivity : LauncherActivity() {

    override fun getTargetIntent(): Intent {
        val intent = Intent(Intent.ACTION_MAIN)
        intent.addCategory("android.intent.category.LAUNCHER")
        return intent
    }
}

这样,就可以得到一个应用启动列表了。


爱奇艺、B站打钱!!

当然,你也可以为MainActivity直接注册以下<intent-filter>:

<intent-filter>
   <action android:name="android.intent.action.MAIN" />
   <category android:name="android.intent.category.LAUNCHER" />
   <category android:name="android.intent.category.HOME" />
   <category android:name="android.intent.category.DEFAULT" />
</intent-filter>

从而让他真的成为你的一个启动器候选项,点击home按键返回桌面时系统会查找注册了这个<intent-filter>的Intent给出一个选择列表,如果安装过第三方桌面应用应该会很熟悉这个弹窗,不过这里如果是厂商rom锁死了启动器的则看不到这个选项,比如目前的miui11不允许使用三方桌面

同样的:文件管理中打开文件的“打开方式列表”也是通过该类型文件对应的<intent-filter>筛选得到的一个Intent列表
例如,想要支持在文件管理器中打开图片,可以注册Actvity的<intent-filter>:

<intent-filter>
  <action android:name="android.intent.action.VIEW"/>
  <category android:name="android.intent.category.DEFAULT"/>
  <data android:mimeType="image/jpeg"/>
</intent-filter>

但是如果想使app出现在分享/发送的列表中,就涉及到另外一个叫做Sharesheet的东西,底层原理上都是一样的,只是再次做了封装,官方文档是翻译版本,已经说的很清楚,就不详细讨论了。

后语:startActivity是怎么通过一个Intent启动起来一个Activity的,以及应用进程怎么启动起来的,我们后期再见。写这篇也算是为了写启动流程做铺垫。PS:如果不懒的话
新鲜出炉啦!!!Activity启动流程(Api29)

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

推荐阅读更多精彩内容