Android 11之后禁止获取第三方应用信息了,比如想获取已安装的所有应用,如果目标版本设置为Android 11,则获取不到了,解决方案就是设置目标版本比Android 11小。如果设置目标版本为Android 11或更高,只能获取指定的应用的信息,在清单文件中声明要获取的应用的包名,如下:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools">
    <!--注:目标版本为Android 11时,要获取第三方应用的信息,必须在清单文件声明需要查看的应用的包名。-->
    <queries>
        <package android:name="cn.android666.example" />
    </queries>
</manifest>

然后在代码中,就可以判断包名为"cn.android666.example"的应用是否安装,如下:

/**
 * 判断指定包名的app是否安装
 * 注:在Android 11版本的时候不允许获取了,不知道把目标版本设置为Android10会不会还给获取呢?答:根据测试,目标版本为22是可以获取的,那目标版本为10应该也可以。
 * 关于Android11包可见性官网连接:
 * https://developer.android.com/about/versions/11/privacy/package-visibility  版本更新的说明
 * https://developer.android.com/training/basics/intents/package-visibility、
 * https://developer.android.com/training/basics/intents/package-visibility-use-cases
 * @param packageName
fun isInstall(packageName: String): Boolean = try {
    mContext.packageManager.getApplicationIcon(packageName)
} catch (e: PackageManager.NameNotFoundException) {
    false
}

更简单的方式是直接添加可以查询所有应用的权限,这样就不需要使用 <queries> 声明了,如下:

<uses-permission android:name="android.permission.QUERY_ALL_PACKAGES"/>

这是一个普通权限,不需要动态申请。

看官方文档描述我以为启动第三方应用也需要添加这个设置,但是实验发现并不需要,代码如下:

button.setOnClickListener {
    val intent = Intent()
    // 参数1为应用的进程包名,参数2为Activity的完整类名
    intent.setClassName("cn.android666.acamera", "cn.android666.acamera.MainActivity")
    startActivity(intent)
}

仔细想想,别人是查询,查询的时候才需要有这个机制,而这里打开第三方应用是填了完整的信息了,已经指定要打开哪个应用的哪个界面了,没有查询操作,所以可以正常打开,即使是使用隐藏意图来打开第三方应用,也是没有查询操作的,也可以正常打开,如下:

val intent = Intent()
intent.action = "abc.def.ghi"
intent.addCategory(Intent.CATEGORY_DEFAULT)
startActivity(intent)

比如我们要通过隐式意图来打开一个播放器来播放我们的音频文件,但有可能这个手机上就没有播放器,任何的播放器都没有,则此时 startActivity(intent) 就会崩溃,所以正确的做法是在启动之前先查询一下是否有能处理此意图的Activity: packageManager.queryIntentActivities(intent, PackageManager.MATCH_DEFAULT_ONLY) ,完整代码如下:

val intent = Intent()
intent.action = "abc.def.ghi"
intent.addCategory(Intent.CATEGORY_DEFAULT) 
val queryIntentActivities: MutableList<ResolveInfo> = packageManager.queryIntentActivities(intent, PackageManager.MATCH_DEFAULT_ONLY)
Toast.makeText(this, "有能处理的吗? ${queryIntentActivities.size}", Toast.LENGTH_SHORT).show()
queryIntentActivities.forEach { Log.i("ABCD", it.activityInfo.name) } // 打印可处理的应用的对应Activity
if (queryIntentActivities.isNotEmpty()) {
    startActivity(intent)
}

此时就用到了查询的方法,就需要在清单文件添加相应的设置了,否则这个查询将返回空集合。查询方法会显示警告以提醒我们进行设置,如下:

android android11 获取已安装app权限 android11无法获取android文件_已安装App


为了简单,我们就用最简单的方式就行了,添加一个权限:

<uses-permission android:name="android.permission.QUERY_ALL_PACKAGES" tools:ignore="QueryAllPackagesPermission" />

然后运行即可。在我的小米11 pro上(Android13)测试,它还会弹出一个框让确认一下,如下:

android android11 获取已安装app权限 android11无法获取android文件_获取已安装_02


es 数组是否包含某元素

一、扩展运算符的应用ES6通过扩展元素符...,好比 rest 参数的逆运算,将一个数组转为用逗号分隔的参数序列console.log(...[1, 2, 3]) // 1 2 3 console.log(1, ...[2, 3, 4], 5) // 1 2 3 4 5 [...document.querySelectorAll('div')] // [<div&gt