• 查询运行在顶部的app包名及类名


    androidActivityManager会在logcat中打印出当前正在显示的app的包名以及类名,注,这是由ActivityManager打印,使用的日志TAG就是ActivityManager,所以我们在过虑信息时不能以自己程序的进程进行过滤,使用ActivityManager作为TAG过虑即可,当然ActivityManager还会显示其它信息,还可以使用Displayed来进行过虑,这样显示的信息就更少一些了,使用新版Logcat示例如下:
    在这里插入图片描述
    可以看到,显示的信息为:

    Displayed org.appspot.loginactivity/.MainActivity: +253ms
    
    • 1

    Displayed 说明了当前正在显示的应用,当前显示的应用包名为org.appspot.loginactivity,类名为.MainActivity,这是一个相对路径的类名,完整路径为org.appspot.loginactivity.MainActivity,这个界面启动的时间为:253ms

    有了这些信息,我们就可以使用Intent来启动这个app了:

    val intent = Intent()
    intent.setClassName("org.appspot.loginactivity", "org.appspot.loginactivity.MainActivity")
    intent.flags = Intent.FLAG_ACTIVITY_NEW_TASK
    startActivity(intent)
    
    • 1
    • 2
    • 3
    • 4

    注:这里的类名必须要使用完整路径的类名org.appspot.loginactivity.MainActivity,不能使用.MainActivity这样的相对路径。

    这有什么用呢?比如公司要我做一个自定义Launcher,只能显示公司的app和一些必要的系统自带app,自定义Launcher显示了需要的app,当点击app图标时,就需要打开该app,此动作由openApp函数实现,如下:

    /** 打开指定包名的app */
    fun openApp(packageName: String) {
        val launcherActivityName = getLauncherActivityNameByPackageName(packageName)
        if (launcherActivityName.isNullOrBlank()) {
            val text = "无法启动应用:$packageName,因为获取不到该应用的启动类"
            Toast.makeText(ContextHolder.getContext(), text, Toast.LENGTH_SHORT).apply { setGravity(Gravity.CENTER, 0, 0) }.show()
            return
        }
        val intent = Intent(Intent.ACTION_MAIN).apply {
            flags = Intent.FLAG_ACTIVITY_NEW_TASK
            component = ComponentName(packageName, launcherActivityName)
        }
        ContextHolder.getContext().startActivity(intent)
    }
    private fun getLauncherActivityNameByPackageName(packageName: String): String? {
        val resolveIntent = Intent(Intent.ACTION_MAIN).apply {
            addCategory(Intent.CATEGORY_LAUNCHER) // 无图标的启动类没有这个Category
            setPackage(packageName)
        }
        val resolveInfoList: List<ResolveInfo> = ContextHolder.getContext().packageManager.queryIntentActivities(resolveIntent, 0)
        val iterator = resolveInfoList.iterator()
        return if (iterator.hasNext()) iterator.next().activityInfo?.name else null
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23

    如上代码,只需要给openApp(packageName: String)函数传入一个应用的包名,即可实现打开该包名对应的app,但是有一个应用我们无法打开,就是“下载”app,这是一个系统应用,比如我们在浏览器下载了一个app,在“下载”app中可以列出浏览器中下载的文件,“下载”app如下:
    在这里插入图片描述
    在这里插入图片描述
    由于通过前面的openApp函数无法打开这个应用,所以此时我们就可以通过查看该应用的包名和类名来启动它,它的启动信息如下:

    Displayed com.android.documentsui/.files.FilesActivity
    
    • 1

    启动该app的代码如下:

    val intent = Intent()
    intent.setClassName("com.android.documentsui", "com.android.documentsui.files.FilesActivity")
    intent.flags = Intent.FLAG_ACTIVITY_NEW_TASK
    startActivity(intent)
    
    • 1
    • 2
    • 3
    • 4
  • 相关阅读:
    【Tools】如何在VSCode上使用C++ 保姆教程
    【备忘】navicat mysql导入导出小经验
    【2022国赛模拟】多边形——计算几何、二分答案、倍增
    01 DevOps 之 Git 命令使用
    信息安全-应用安全-蚂蚁集团软件供应链安全实践
    Seata 源码篇之AT模式启动流程 - 中 - 03
    加强堆的应用:topK问题
    ANACONDA的进阶理解和思考
    Linux安装JDK和Tomcat
    《概率的哲学理论 Philosophical Theories of Probability》读书笔记 吉利斯Gillies
  • 原文地址:https://blog.csdn.net/android_cai_niao/article/details/128141808