获取所有用户的应用列表

如果我想检索当前用户的所有应用程序的应用程序信息列表,我可以运行:

PackageManager pkgmanager = ctx.getPackageManager();
List<ApplicationInfo> installedApps = pkgmanager.getInstalledApplications(PackageManager.GET_META_DATA);

但是我正在寻找一种方法来为每个用户获取这些列表,而不仅仅是当前用户。顺便说一句,我正在使用的应用程序具有root权限!

据我所知,可以像这样检索用户ID:

List<Integer> userIds = new ArrayList<>();
PackageManager pkgmanager = ctx.getPackageManager();
final UserManager um = (UserManager) ctx.getSystemService(Context.USER_SERVICE);
List<UserHandle> list = um.getUserProfiles();

for (UserHandle user : list) {
    Matcher m = p.matcher(user.toString());
    if (m.find()) {
        int id = Integer.parseInt(m.group(1));
            userIds.add(id);
    }
}

现在我正在寻找这样的东西:

for (int i=0; i<userIds.size(); i++) {
    Integer userId = userIds.get(i)
    List<ApplicationInfo> installedApps = pkgmanager.getInstalledApplicationsByUserId(userId, PackageManager.GET_META_DATA);
}

显然不存在。getInstalledApplicationsByUserId

起初,我认为getPackagesForUid可以解决这个问题,但事实证明,Linux意义上的“用户”和用户配置文件之间存在差异。通常,出于隔离目的,每个Android应用程序都在自己的用户下运行。但是,可以在同一用户下运行两个应用程序,以便它们可以轻松访问彼此的数据。 仅返回在给定用户 ID 下运行的所有应用的名称,该 ID 通常正好是一个。除了“用户”之外,还有“用户配置文件”,这是我希望该方法所指的。也许我也应该写而不是在我的代码中。getPackagesForUiduserProfileIduserId

编辑:使用adb shell,我可以像这样检索应用程序ID:

# Get user profile IDs
USER_PROFILE_IDS="$(pm list users | grep UserInfo | cut -d '{' -f2 | cut -d ':' -f1)"

# Iterate over user profile IDs
while read -r USER_PROFILE_ID ; do
    # List the packages for each user profile by ID
    PACKAGE_IDS_FOR_THIS_USER="$(pm list packages --user "$USER_PROFILE_ID" | cut -d ':' -f2)"
    echo "#######################################################################"
    echo "The user with id $USER_PROFILE_ID has the following packages installed:"
    echo "$PACKAGE_IDS_FOR_THIS_USER"
done <<< "$USER_PROFILE_IDS"

但那是Bash而不是Java...

Edit2:如果我错了,请纠正我(这是我第一次用Java编写代码),但似乎没有Java API来做到这一点。因此,执行此操作的唯一方法是使用 shell。这就是我想到的:

import com.stericson.rootshell.RootShell;
import com.stericson.rootshell.execution.Command;
import com.stericson.rootshell.execution.Shell;
import com.stericson.roottools.RootTools;
...
public final class Api {
    ...
    /**
     * @param ctx application context (mandatory)
     * @return a list of user profile ids
     */
    private static List<Integer> getUserIds(Context ctx) {
        List<Integer> userIds = new ArrayList<>();
        PackageManager pkgmanager = ctx.getPackageManager();
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
            //this code will be executed on devices running ICS or later
            final UserManager um = (UserManager) ctx.getSystemService(Context.USER_SERVICE);
            List<UserHandle> list = um.getUserProfiles();

            for (UserHandle user : list) {
                Matcher m = p.matcher(user.toString());
                if (m.find()) {
                    int id = Integer.parseInt(m.group(1));
                    //if (id > 0) {
                        userIds.add(id);
                    //}
                }
            }
        } else {
            userIds.add(0);
        }
        return userIds;
    }

    /**
     * @param ctx application context (mandatory)
     * @return a list of user profile ids
     */
    private static List<String> getPackageIdsByUserProfileId(Integer userId) {
        List<String> packageIds = new ArrayList<>();
        Command command = new Command(0, "pm list packages --user " + userId + " | cut -d ':' -f2 ")
        {
            @Override
            public void commandOutput(int id, String line) { 
                packageIds.add(line);
                super.commandOutput(id, line);
            }
        };
        Shell shell = RootTools.getShell(true);
        shell.add(command);

        while (!command.isFinished()) {
            Thread.sleep(100);
        }
        return packageIds;
    }
...
    /**
     * @param ctx application context (mandatory)
     * @return a list of applications
     */
    public static List<PackageInfoData> getApps(Context ctx, GetAppList appList) {
        List<Integer> userIds = getUserIds();
        for (int i=0; i<userIds.size(); i++) {
            Integer userId = userIds.get(i)
            List<String> packageIds = getPackageIdsByUserProfileId(userId)
        }
        ...
    }
}

但我不知道这是否接近实际有效的东西。除此之外,我只获得每个用户配置文件的软件包ID(“com.whatsapp”等),但我想得到一个AppplicationInfo列表,就像返回它一样。我只是想不出一个好方法来做到这一点。也许可以加载包清单,然后以某种方式创建应用程序信息实例?getInstalledApplications

编辑3:我想我找到了pm可执行文件的源代码

奇怪的是,我找不到一个提到国旗的地方。--user

代码中最相关的部分是:

import android.os.ServiceManager;
...
mPm = IPackageManager.Stub.asInterface(ServiceManager.getService("package"));
...
int getFlags = 0;
...
final List<PackageInfo> packages = getInstalledPackages(mPm, getFlags);

该方法只需调用:getInstalledPackagesmPm.getInstalledPackages

@SuppressWarnings("unchecked")
private List<PackageInfo> getInstalledPackages(IPackageManager pm, int flags)
        throws RemoteException {
    final List<PackageInfo> packageInfos = new ArrayList<PackageInfo>();
    PackageInfo lastItem = null;
    ParceledListSlice<PackageInfo> slice;
    do {
        final String lastKey = lastItem != null ? lastItem.packageName : null;
        slice = pm.getInstalledPackages(flags, lastKey);
        lastItem = slice.populateList(packageInfos, PackageInfo.CREATOR);
    } while (!slice.isLastSlice());
    return packageInfos;
}

这让我比以前有更多的问题。首先,我想知道我是否不能只导入类。其次,我想知道我怎么能告诉它返回特定用户配置文件的包,或者这是否是第一个正确的源代码。最后,我想知道我是否需要root权限才能使用它。毕竟,仅对 和 执行检查。com.android.commands.pmif (Process.myUid() != ROOT_UID)runRemoveProfilerunCreateProfilerunListProfiles

编辑4:我无法找到服务的源代码。我只能找到这个文件:.它包含所有程序包(对于所有用户配置文件)的一些基本程序包信息,但它既不包含实际的应用名称,也不包含有关它们所属的用户配置文件的信息。package/data/system/packages.xml

Edit5:我想我找到了包服务源代码。我认为这种方法很重要:

public ParceledListSlice<PackageInfo> getInstalledPackages(int flags, int userId)

不幸的是,我只是不理解代码。对我来说,看起来这些软件包以某种方式来自.该变量在代码注释中解释如下:mSettings.mPackages

{@link #mPackages} 用于保护所有内存中解析的包详细信息和其他相关状态。它是一个细粒度的锁,只能暂时持有,因为它是系统中最具争议的锁之一。

编辑6:我刚刚在那个文件中发现了另一个更有趣的方法:

public ParceledListSlice<ApplicationInfo> getInstalledApplications(int flags, int userId)

我不知道ParceredListSlice是什么,但它说我假设这真的接近我想要的格式。但是,我仍然完全不知道如何访问该功能。<ApplicationInfo>List<ApplicationInfo>


答案 1

我在这里找到了代码,以及在这里实现的接口的代码。看起来它在内部使用某种形式的数组,所以你应该能够迭代它。我实际上不能说这对你是否有帮助,因为我对这些库并不熟悉,但我希望这是一个起点。祝你好运!ParceledListSliceParcelable


答案 2

您可以应用您在第一个EDIT和EDIT2中提到的方法并使用Android的shell,因为您提到您的设备具有root权限。

顺便说一句,我不熟悉您正在使用的“stericson”中的库,但我同意利用现有库来使您的生活更轻松;我正在使用libsuperuser。否则,您可以自己编写代码以使用 Process 等运行 shell 命令(但是在处理错误输出、关闭对象等时需要考虑很多事项)。

private List<String> listAllInstalledApps(){
    String user = "0"; //can modify to iterate over all users 
    if (Shell.SU.available()) {
        //grab user installed package names
        List<String> output =
            Shell.SU.run("pm list packages --user " + user + " -3");
        StringBuilder csvBuilder = new StringBuilder();

        for(String item : output){
            csvBuilder.append(item);
            csvBuilder.append(", ");
        }
        Log.info(TAG, "Obtained installed apps: " + csvBuilder.toString());
        return output;
    }
    return new ArrayList<>();
}

当然,您可以根据需要使用其他参数来表示 pm,例如查看文档-e, -d

提取包名称后,使用 dumpsys 获取所有必需的信息(包含在 中)。ApplicationInfo

//package names extracted from List<String> you returned earlier
for( String name : packageNames ){
    List<String> appInfo =
            Shell.SU.run("dumpsys package " + name);
    //search appInfo for info you need e.g. grantedPermissions, targetSdk...
}

根据需要创建 ApplicationInfo 对象(如果有专门需要此对象类型的下游代码)

ApplicationInfo info = new ApplicationInfo();
info.uid = uid;
info.processName = processName;
info.className = classname;
info.packageName = packageName;
info.minSdkVersion = minSdkVersion;
//etc...

(类的字段是公共的)。这也是一个包含字段的对象,它包含安装时间等的详细信息,这些详细信息也是您可以从dupsys输出中提取的详细信息。因此,您可以根据需要创建这些数组,并可以使用新对象进行填充。ApplicationInfoPackageInfoApplicationInfoApplicationInfo


推荐