《Android编程权威指南》之深入学习intent和任务

《Android编程权威指南》第 23 章,本章将创建新应用啦,叫 NerdLauncher,然后技术点是关于隐式 intent 和 intent 过滤器。本章应用呢,将会展示设备上的其他应用,还可以启动其他应用。

一、创建 NerdLauncher 项目

创建项目,添加 RecyclerView 用于显示 App 列表。

<?xml version="1.0" encoding="utf-8"?>
<androidx.recyclerview.widget.RecyclerView xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/recycler_view"
    android:layout_width="match_parent"
    android:layout_height="match_parent" />
class MainActivity : AppCompatActivity() {

    private lateinit var mBinding: ActivityMainBinding

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        mBinding = ActivityMainBinding.inflate(layoutInflater)
        setContentView(mBinding.root)

        mBinding.recyclerView.layoutManager = LinearLayoutManager(this)
    }
}

二、解析隐式 intent

PackageManager 可用来获取所有可启动主 activity。可启动主 activity 都带有包含 MAIN 操作和 LAUNCHER 类别的 intent 过滤器。如下所示:

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

新增 setupAdapter() 函数,并在 onCreate() 中调用:

   private fun setupAdapter(){
       val startupIntent = Intent(Intent.ACTION_MAIN).apply {
           addCategory(Intent.CATEGORY_LAUNCHER)
       }
       val activities = packageManager.queryIntentActivities(startupIntent,0)
       Log.i(TAG, "Found ${activities.size} activities")
   }

上述代码就有创建操作设为 ACTION_MAIN、类别为 CATEGORY_LAUNCHER 的隐式 intent。

PackageManager.queryIntentActivities(Intent, Int) 会返回包含所有activity(有匹配目标 intent 的过滤器)的 ResolveInfo 信息。这里参数 0 表示不打算修改查询结果。

activities

接下来需要展示列表,使用 activity 标签名「应用名」。首先,使用ResolveInfo.loadLabel(PackageManager) 函数,对 ResolveInfo 对象中的 activity 标签按首字母排序。

在上述方法下面加入代码:

activities.sortWith(Comparator { a, b -> 
            String.CASE_INSENSITIVE_ORDER.compare(
                a.loadLabel(packageManager).toString(),
                b.loadLabel(packageManager).toString()
            )
        })

然后,定义一个 ViewHolder 用来显示 activity 标签名。使用成员变量存储 ResolveInfo 。

    private class ActivityHolder(itemView: View):RecyclerView.ViewHolder(itemView){
        
        private val tvName = itemView as TextView
        private lateinit var resolveInfo:ResolveInfo
        
        fun bindActivity(resolveInfo: ResolveInfo){
            this.resolveInfo = resolveInfo
            val packageManager = itemView.context.packageManager
            val appName = resolveInfo.loadLabel(packageManager).toString()
            tvName.text = appName
        }
    }

接下来实现 RecyclerView.Adapter:

    private class ActivityAdapter(val activities:List<ResolveInfo>):RecyclerView.Adapter<ActivityHolder>(){
        override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): ActivityHolder {
                val layoutInflater = LayoutInflater.from(parent.context)
                val view =layoutInflater.inflate(android.R.layout.simple_list_item_1,parent,false)
                 return ActivityHolder(view)
            }

        override fun onBindViewHolder(holder: ActivityHolder, position: Int) {
            val resolveInfo = activities[position]
            holder.bindActivity(resolveInfo)
        }

        override fun getItemCount(): Int {
            return activities.size
        }
    }

最后,把 adapter 实例配置给 RecyclerView ,在 setupAdapter 末尾处添加:

 mBinding.recyclerView.adapter = ActivityAdapter(activities)

运行结果:

result

三、在运行时创建显式intent

接下来做点击列表,用显示 intent 启动对应的 activity 啦。

要创建启动 activity 的显式 intent,就需要从 ResolveInfo 对象中获取 activity 的包名与类名。

更新 ActivityHolder 类实现一个点击监听器,并从 activityInfo 中获取必要信息,创建一个显示 intent 去启动目标 activity。

    private class ActivityHolder(itemView: View) : RecyclerView.ViewHolder(itemView),
        View.OnClickListener {
       ...
        init {
            tvName.setOnClickListener(this)
        }
       ...
        override fun onClick(v: View) {
            val activityInfo = resolveInfo.activityInfo
            val intent = Intent(Intent.ACTION_MAIN).apply {
                setClassName(activityInfo.applicationInfo.packageName, activityInfo.name)
            }
            val context = v.context
            context.startActivity(intent)
        }
    }

然后运行项目,点击列表 item,就可以跳转到相应的 App 了。

四、任务与回退栈

Android 使用任务来跟踪应用运行的状态。

任务是一个 activity 栈。栈底部的 activity 通常称为基 activity。栈顶的 activity 用户能看得到。按回退键,栈顶 activity 会弹出栈外。如果用户看到的是基 activity,按回退键,系统就会回到主屏幕。

有关 Activity 任务栈,可以看看扔物线的视频,讲的很详细易懂,一看就懂,不过还是需要多看几遍,思考一下,最好再实践一下理解更深刻:

https://www.bilibili.com/video/BV1CA41177Se?spm_id_from=333.999.0.0

当前应用去打开其他的 App,其他 App 的启动 Activity 都是运行在自身的任务栈中的,为了在启动新 activity 时启动新任务,需要为 intent 添加一个标志:

addFlags(Intent.FLAG_ACTIVITY_NEW_TASK)

FLAG_ACTIVITY_NEW_TASK 标志控制每个 activity 仅创建一个任务。如果那个被打开的 activity 没有任务栈就会创建一个新的任务栈,如果已经有了一个运行的任务,Android 就会自动切回到那个任务,就不再创建新的任务了。

五、用 NerdLauncher 当主屏幕

这里实践把 NerdLauncer 应用成 Android 主界面(home screen),「我们的桌面实际上也是 Android 系统中的一个应用,显示着我们安装的各个 App,给我们启动 App 的入口」。

只需要修改 NerdLauncherActivity 的类别即可,打开 manifests/AndroidManifest.xml:

        <activity
            android:name=".MainActivity"
            android:exported="true">
            <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>
        </activity>

不过我这里实践按主屏幕键没有看到书中的那个框,可能跟 Android 版本有关。下次再研究一下。

六、深入学习:进程与任务

『进程』 是操作系统创建的、供应用对象生存以及应用运行的地方。通常会拥有由操作系统管理着的一些系统资源,比如内存、网络端口以及打开的文件等。拥有至少一个执行线程,Android 系统中,每个进程都需要一个 虚拟机 来运行。

Android 4.4(KitKat)之前,Dalvik 是 Android 操作系统使用的进程虚拟机。进程只要一启动,就会有一个 Dalvik 虚拟机新实例跳出来收留它。不过,自Android 5.0(Lollipop)开始,Android运行时(ART)取代了 Dalvik,已成为公认的进程虚拟机。

之前的应用 CriminalIntent,联系人应用虽然是 CriminalIntent 打开的,联系人列表 activity 会被加入到 CriminalIntent 应用任务中,可是联系人 activity 实例实际是在联系人应用进程的内存空间创建的,而且也是在该应用进程里的虚拟机上运行的。如图所示:

任务与进程一对多的关系

接下来再去试验一个过程,在 CriminalIntent 中进入联系人列表,按 Home 回到桌面,从桌面去启动联系人应用,从联系人列表中选取联系人或添加联系人。这个过程,系统会在联系人应用进程中创建新的联系人列表 activity 和联系人明细界面实例。也会创建联系人应用新任务。这个新任务会引用联系人列表和联系人明细界面 activity 实例,如图所示:

进程对多个任务

理解完本章,我们应该知道,Google Play 商店中一些自称为任务终止器的应用,实际上都是进程终止器。这些应用会“杀掉”某个进程,这表明,它们可能正在销毁其他应用任务引用的 activity。

七、深入学习:并发文档

并发文档(concurrent document):在Android Lollipop(API 级别 21)上引入,可以为运行的应用动态创建任意数目的任务。

Google Drive 是并发文档概念应用的最好实例。用户可以用它打开并编辑多份文档。从概览屏可以看到,这些文档编辑activity都处在独立的任务中。

多个 Google Drive 任务

如果需要应用启动多个任务,可给 intent 打上 Intent.FLAG_ACTIVITY_NEW_DOCUMENT 标签,再调用startActivity(...)函数;或在 manifest 中,为 activity 设置如下 documentLaunchMode:

android:documentLaunchMode="intoExisting"

这样,一份文档只会对应一个任务。(如果发送带有和已存在任务相同数据的intent,系统就不会再创建新任务。)如果无论如何都想创建新任务,那就给intent 同时打上 Intent.FLAG_ACTIVITY_NEW_DOCUMENT 和 Intent.FLAG_ACTIVITY_MULTIPLE_TASK 标签,或把 manifest 中的 documentLaunchMode 属性值改为 always。

自行实践~ O(∩_∩)O哈哈~

八、挑战练习:应用图标

给 NerdLauncher 应用中显示的所有应用添加图标。

简单的呢,就是在给列表设置文字内容的下面添加代码:

            val appIcon = resolveInfo.loadIcon(packageManager)
            appIcon.setBounds(0, 0, appIcon.minimumWidth, appIcon.minimumHeight)
            tvName.setCompoundDrawables(appIcon, null, null, null)

运行效果「丑丑的」:

result

其他

ResolveInfo 还可以获取其他信息,详情介绍请参考:

https://developer.android.com/reference/android/content/pm/ResolveInfo

NerdLauncher 项目 Demo 地址:

https://github.com/visiongem/AndroidGuideApp/tree/master/NerdLauncher

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

推荐阅读更多精彩内容