Android 基础 之 三种方法尝试如何完整的获取到用户已安装应用列表
Android 如何完整的获取到用户已安装应用列表目录Android 如何完整的获取到用户已安装应用列表方案1方案2方案3结论接到产品经理的预研需求,说希望获取用户已安装应用列表。这个问题应该不难,只要是要把相关的知识点整理和验证一下。对于获取用户已安装应用列表,我个人是很熟悉的,因为我的华为手机上,手机管家天天会在通知栏弹出”xxx应用尝试获取用户已安装应用列表被禁止”。所以,很明显,跟权限是有
Android 三种方法尝试如何完整的获取到用户已安装应用列表
目录
Android 三种方法尝试如何完整的获取到用户已安装应用列表
接到产品经理的预研需求,说希望获取用户已安装应用列表。这个问题应该不难,只要是要把相关的知识点整理和验证一下。
对于获取用户已安装应用列表,我个人是很熟悉的,因为我的华为手机上,手机管家天天会在通知栏弹出”xxx应用尝试获取用户已安装应用列表被禁止”。所以,很明显,跟权限是有关系的。于是,我尝试去查找到底是manifest清单中的哪一个use-permission引起。结果,找了很久,翻了很久,并没有哪个权限对已安装的应用列表负责。
但奇怪的是,我的手机上几乎全部的软件都声明了这个权限。于是,尝试去求助其他组员,咨询了几个,不少人一脸懵逼的表示这是个什么玩意。在他们的手机上压根就没有见过这个东西。
在写demo验证的过程中,发现非常简单的一个demo,居然也声明使用了该权限。 一开始怀疑,难道是检测到了相关代码自动申请了权限?发现全部注释后还是会声明。 后来,将清单文件中的唯一的访问Internet权限去掉,这样才正常。
所以,得出了一个结论就是,国内部分厂商比如华为、oppo,他们将”获取用户已安装应用列表”的权限暴露给了用户,让用户可以自由决定允许或者禁止应用访问该信息。同时,这个权限类似于附加的默认权限,一旦app声明了任何权限,那么”获取用户已安装应用列表”的权限也会被附加进来。但这个权限也不是太敏感,所以对于用户是无感知的。这里的无感知指的是不会在应用中去主动让我们弹窗申请权限,手机管家弹出的通知不算。
好吧,说了这么多,看一下过程中的3种方案。
方案1
private void getAppList() {
PackageManager pm = getPackageManager();
// Return a List of all packages that are installed on the device.
List<PackageInfo> packages = pm.getInstalledPackages(0);
for (PackageInfo packageInfo : packages) {
// 判断系统/非系统应用
if ((packageInfo.applicationInfo.flags & ApplicationInfo.FLAG_SYSTEM) == 0) // 非系统应用
{
System.out.println("MainActivity.getAppList, packageInfo=" + packageInfo.packageName);
} else {
// 系统应用
}
}
}
此方法在华为、oppo手机上,把权限禁止后,就不能正确获取到已安装应用列表了。
方案2
考虑到方案1受权限的影响,于是考虑用adb命令去获取已安装的应用列表。
命令:adb shell pm list package -3
上面的命令可以获取到手机上已安装的第3方应用列表,去掉-3这个参数可以获取到全部的应用列表。本来对这个方案抱挺大的期望的,但是最终发现在oppo手机上,如果禁止了获取已安装应用列表的权限,那么结果就会受到影响,无奈又不行。
小插曲:在代码调用命令行过程中遇到个坑,
private void runCommand() {
try {
Process process = Runtime.getRuntime().exec("adb shell pm list package -3");
BufferedReader bis = new BufferedReader(new InputStreamReader(process.getInputStream()));
String line = null;
while ((line = bis.readLine()) != null) {
System.out.println("MainActivity.runCommand, line=" + line);
}
} catch (IOException e) {
System.out.println("MainActivity.runCommand,e=" + e);
}
}
用代码去执行了 adb shell pm list package -3的命令,发现一直报IOException,最终耗费一定的时间,定位到问题。我们使用adb shell是因为手机跟pc要连接,但是在手机上运行时,其实不用加adb shell,直接执行”pm list package -3”即可。
方案3
采用getPackageManager().queryIntentActivities(intent,PackageManager.MATCH_ALL)去查询是否有符合指定意图的Activity,从而判断是否安装了某应用。
该方法返回的是ResolveInfo列表,而ResolveInfo包含的是IntentFilter信息。
/***
* 使用 ResolveInfo 獲取 應用列表信息
* @return
*/
private List<AppInfo> loadAllApplication() {
PackageManager manager = getActivity().getPackageManager();
List<AppInfo> myAppInfos = new ArrayList<AppInfo>();
mLocalInstalledApps_Tmp = new ArrayList<AppInfo_Tmp>();
Intent intent = new Intent(Intent.ACTION_MAIN, null);
intent.addCategory(Intent.CATEGORY_LAUNCHER);
List<ResolveInfo> availableActivities = manager.queryIntentActivities(intent, 0);
int no=0;
for (ResolveInfo ri : availableActivities) {
ApplicationInfo applicationInfo = ri.activityInfo.applicationInfo;
if((applicationInfo.flags & (ApplicationInfo.FLAG_UPDATED_SYSTEM_APP | ApplicationInfo.FLAG_SYSTEM)) > 0) {
// It is a system app
} else {
Log.i(TAG, "loadAllApplication: applicationInfo.packageName " + applicationInfo.packageName);
Log.i(TAG, "loadAllApplication: applicationInfo.label " + applicationInfo.loadLabel(manager));
no ++;
AppInfo myAppInfo = new AppInfo();
AppInfo_Tmp myAppInfo_Tmp = new AppInfo_Tmp();
// app packageName
myAppInfo.setPackageName(applicationInfo.packageName);
myAppInfo_Tmp.setPackageName(applicationInfo.packageName);
// app appName
myAppInfo.setAppName(applicationInfo.loadLabel(manager).toString());
if (ri.loadIcon(manager) ==null){
Log.i(TAG, "loadAllApplication: ri.loadIcon(manager) ==null");
}else {
myAppInfo_Tmp.setAppIcon(ri.loadIcon(manager));
}
myAppInfos.add(myAppInfo);
mLocalInstalledApps_Tmp.add(myAppInfo_Tmp);
}
}
//Toast.makeText(getActivity(),"App no:"+no,Toast.LENGTH_SHORT).show();
Log.i(TAG, "loadAllApplication: App no:"+no);
return myAppInfos;
}
以下结论都经过demo的验证:
- 清单文件的声明中必须包含IntentFilter信息,queryIntentActivities方法才能查找到。
-
IntentFilter中不能包含data信息,如果有定义,则查找不到信息了,连启动的Activity都找不到。这里暂时没有去查看The Fucking Source Code。
-
添加 android:exported与否对于此方法的结果没有影响
-
在华为等对应用安装列表有权限控制的手机上,采用隐式的Intent获取不到正确的信息,就连每个应用的启动Activity都获取不到。
-
显示的Intent则不受权限的影响,均可以获取到。
结论
采用第三种方案,用显示的Intent(一般指定包名或者类名)去查询是否安装了某应用在各个厂商各个系统的手机上是可行的,但是只能获取到指定的,而不是全部。
转载地址:https://blog.csdn.net/q384415054/article/details/72972405
开放原子开发者工作坊旨在鼓励更多人参与开源活动,与志同道合的开发者们相互交流开发经验、分享开发心得、获取前沿技术趋势。工作坊有多种形式的开发者活动,如meetup、训练营等,主打技术交流,干货满满,真诚地邀请各位开发者共同参与!
更多推荐
所有评论(0)