4

背景

在 Android O 之前,为了获得应用程序大小,您必须对其具有特殊权限,并使用隐藏的 API。

问题

这样的解决方案将不再有效,而是 Google 提供了一个粒度较小的官方 API:queryStatsForPackage,它返回StorageStats对象。

但是,我找不到任何有关如何使用它的信息。

与只需要应用程序包名的隐藏 API 不同,此 API 还需要“String volumeUuid”和“UserHandle user”。

问题

  1. 我如何提供这些参数?
  2. 他们的意思是什么?每个 volumeUuid 是否引用不同的存储(例如,内置存储和真正的 sd 卡),并且 UserHandle 是针对每个用户的?
  3. 如果#2 是正确的,我真的应该为每个可能的值调用它们,以获得应用程序占用的实际总存储空间吗?如果是这样,怎么做?
4

3 回答 3

11

好的,我已经了解了如何使用新的 API。我准备了一个小样本,获取有关卷存储和特定应用程序的一些信息(这次是 Play 商店,但您可以根据需要进行更改):

public class MainActivity extends AppCompatActivity {

    public static final String PACKAGE_NAME = "com.android.vending";

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
    }

    @Override
    protected void onResume() {
        super.onResume();
        if (!hasUsageStatsPermission(this))
            startActivityForResult(new Intent(Settings.ACTION_USAGE_ACCESS_SETTINGS).addFlags(Intent.FLAG_ACTIVITY_NO_HISTORY), 1);
        else {
            final Context context = this;
            AsyncTask.execute(new Runnable() {
                @TargetApi(VERSION_CODES.O)
                @Override
                public void run() {
                    @SuppressLint("WrongConstant") final StorageStatsManager storageStatsManager = (StorageStatsManager) getSystemService(Context.STORAGE_STATS_SERVICE);
                    final StorageManager storageManager = (StorageManager) getSystemService(Context.STORAGE_SERVICE);
                    final List<StorageVolume> storageVolumes = storageManager.getStorageVolumes();
                    final UserHandle user = android.os.Process.myUserHandle();
                    for (StorageVolume storageVolume : storageVolumes) {
                        final String uuidStr = storageVolume.getUuid();
                        final UUID uuid = uuidStr == null ? StorageManager.UUID_DEFAULT : UUID.fromString(uuidStr);
                        try {
                            Log.d("AppLog", "storage:" + uuid + " : " + storageVolume.getDescription(context) + " : " + storageVolume.getState());
                            Log.d("AppLog", "getFreeBytes:" + Formatter.formatShortFileSize(context, storageStatsManager.getFreeBytes(uuid)));
                            Log.d("AppLog", "getTotalBytes:" + Formatter.formatShortFileSize(context, storageStatsManager.getTotalBytes(uuid)));
                            Log.d("AppLog", "storage stats for app of package name:" + PACKAGE_NAME + " : ");

                            final StorageStats storageStats = storageStatsManager.queryStatsForPackage(uuid, PACKAGE_NAME, user);
                            Log.d("AppLog", "getAppBytes:" + Formatter.formatShortFileSize(context, storageStats.getAppBytes()) +
                                    " getCacheBytes:" + Formatter.formatShortFileSize(context, storageStats.getCacheBytes()) +
                                    " getDataBytes:" + Formatter.formatShortFileSize(context, storageStats.getDataBytes()));
                        } catch (NameNotFoundException e) {
                            e.printStackTrace();
                        } catch (IOException e) {
                            e.printStackTrace();
                        }
                    }
                }
            });

        }
    }

    @TargetApi(VERSION_CODES.M)
    public static boolean hasUsageStatsPermission(Context context) {
        //http://stackoverflow.com/a/42390614/878126
        if (VERSION.SDK_INT < VERSION_CODES.LOLLIPOP)
            return false;
        AppOpsManager appOps = (AppOpsManager) context.getSystemService(Context.APP_OPS_SERVICE);
        final int mode = appOps.checkOpNoThrow(AppOpsManager.OPSTR_GET_USAGE_STATS, android.os.Process.myUid(), context.getPackageName());
        boolean granted = false;
        if (mode == AppOpsManager.MODE_DEFAULT)
            granted = (context.checkCallingOrSelfPermission(android.Manifest.permission.PACKAGE_USAGE_STATS) == PackageManager.PERMISSION_GRANTED);
        else
            granted = (mode == AppOpsManager.MODE_ALLOWED);
        return granted;
    }
}

清单文件应该有这个:

<uses-permission
    android:name="android.permission.PACKAGE_USAGE_STATS"
    tools:ignore="ProtectedPermissions"/>

编辑:请注意,UUID.fromString(uuidStr)SD卡的部分可能会失败,谷歌的原因是这样的:

Android 无法为公共卷(SD 卡等)维护配额式统计信息。应用程序需要自己构建大小计算逻辑。

可悲的是,在 Android 11 上,由于到达“Android”文件夹的新限制,即使这也是不可能的。在这里要求改变它:

@t0m 但即使他们写了什么,官方也不是不可能按照他们所说的去做,因为存储权限在 Android 11 上受到了非常严格的限制,不允许您正确访问 Android 文件夹。要求在这里解决这个问题:

请考虑主演。

于 2017-06-22T19:51:47.087 回答
1

获取 UUID 的另一种方法。

要获取 UUID,您还可以使用 StorageManager.getUuidForPath(File path) 函数。至于路径,使用 Context.getFilesDir() 和 Context.getExternalFilesDirs() 来获取应用数据将被保存到的所有可能的目录。

请注意,不同的路径可能返回相同的 UUID,因此您必须使用哈希集来收集唯一的 UUID

于 2017-06-30T01:01:37.210 回答
0

将 null 作为 VolumeUuid 传递,以获取默认内部存储的值,或通过 StorageManager 系统服务获取特定卷。

您可以通过 Process.myUserHandle() 获得自己的 UserHandle。当您在设备上为某人提供访客登录或设备上安装了 Android for Work 配置文件时,系统的其他用户就会出现。我认为没有一种简单的方法可以枚举所有有权访问该设备的用户。

于 2017-04-21T13:19:23.163 回答