Android 获取应用总大小

Android 获取应用总大小

本文的所有代码都是用kotlin写的,但是原理都是一样的。一个应用的总大小,包括应用大小、用户数据和缓存。这些数据我们该怎样获取了,下面我们一起来看一看!

Android 8.0以前的获取方法

在Android 8.0以前获取应用的总大小,google没有提供对应API,但是我们可以通过反射来获取,代码如下:

fun getAppTotalsize(context: Context, pkgName: String) {
    val mPackManager: PackageManager = context.packageManager
    //反射获取PackageManager内部getPackageSizeInfo方法
    val method = PackageManager::class.java.getMethod("getPackageSizeInfo", String::class.java, IPackageStatsObserver::class.java)
    var appSizeL: Long
    method.invoke(mPackManager, pkgName, object : IPackageStatsObserver.Stub() {
        @Throws(RemoteException::class)
        override fun onGetStatsCompleted(pStats: PackageStats, succeeded: Boolean) {
            //应用的总大小等于缓存大小加上数据大小再加上应用的大小
            appSizeL = pStats.cacheSize + pStats.dataSize + pStats.codeSize        
    })
}

上面的代码还是很简单的,但是有的同学可能会不理解,IPackageStatsObserver.Stub() 是怎么来的,其实就是IPC,上面的实现需要有两个aidl文件IPackageStatsObserver.aidl和PackageStats.aidl,他们的内容如下:

IPackageStatsObserver.aidl
package android.content.pm;
import android.content.pm.PackageStats;
interface IPackageStatsObserver {
    oneway void onGetStatsCompleted(in PackageStats pStats, boolean succeeded);
}
PackageStats.aidl
package android.content.pm;
parcelable PackageStats;

这个IPC过程是非常简单的,这里就不多做说明了,如果不知道IPC的同学可以自己去看看,在此我推荐一篇博客android跨进程通信(IPC):使用AIDL ,以上就是Android 8.0以前获取应用总大小的方法了,下面我们来看看8.0以后的获取方法。

Android 8.0以后的获取方法

在Android 8.0以后,google出于安全的考虑,废除了以前的方法,但是为我们提供了新的API,我们可以使用StorageStatsManager 类来获取应用大小、用户数据和缓存,从而得到应用的总大小。但是在使用StorageStatsManager 类获取时,我们需要android.permission.PACKAGE_USAGE_STATS 权限。所以在Android 8.0以后要获取应用的总大小,我们必须先申请权限,用户授权了之后才能获取。首先,我们要判断是否开启了android.permission.PACKAGE_USAGE_STATS 权限,代码如下:

/**
* 判断是否有#PACKAGE_USAGE_STATS#的权限
*/
@RequiresApi(Build.VERSION_CODES.M)
fun checkUsageStats(activity: Activity): Boolean {
    val granted: Boolean
    val appOps = activity.getSystemService(Context.APP_OPS_SERVICE) as AppOpsManager
    val mode = appOps.checkOpNoThrow(AppOpsManager.OPSTR_GET_USAGE_STATS, android.os.Process.myUid(), activity.packageName)
    if (mode == AppOpsManager.MODE_DEFAULT) {
        granted = activity.checkCallingOrSelfPermission(android.Manifest.permission.PACKAGE_USAGE_STATS) ==
                    PackageManager.PERMISSION_GRANTED
    } else {
        granted = mode == AppOpsManager.MODE_ALLOWED
    }
    return granted
}

然后,如果没有开启,我们需要引导用户去开启权限,该权限需要跳转到设置界面用户必须手动开启,跳转代码如下:

fun openUsagePermissionSetting(context: Context) {
    val intent = Intent(Settings.ACTION_USAGE_ACCESS_SETTINGS)
    intent.flags = Intent.FLAG_ACTIVITY_NEW_TASK
    intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TASK)
    context.startActivity(intent)
}

最后,获取到了权限,我们就可以开始获取应用的总大小了,代码如下:

fun getAllAppTotalsizeO(context: Context, pkgName: String){
    val storageStatsManager: StorageStatsManager = context.getSystemService(Context.STORAGE_STATS_SERVICE)
as StorageStatsManager
    val storageManager: StorageManager = context.getSystemService(Context.STORAGE_SERVICE) as StorageManager
    //获取所有应用的StorageVolume列表
    val storageVolumes: List<StorageVolume> = storageManager.storageVolumes
    for (item in storageVolumes) {
        val uuidStr = item.uuid
        val uuid: UUID = if (uuidStr == null) StorageManager.UUID_DEFAULT else UUID.fromString(uuidStr)
        //通过包名获取uid
        val uid = getUid(context, pkgName)
        val storageStats = storageStatsManager.queryStatsForUid(uuid, uid)
        //获取到App的总大小
        appSizeL = storageStats.appBytes + storageStats.cacheBytes + storageStats.dataBytes
    }
}

/**
* 根据应用包名获取对应uid
*/
fun getUid(context: Context, pakName: String): Int {
    val pm = context.packageManager
    try {
        val ai = pm.getApplicationInfo(pakName, PackageManager.GET_META_DATA)
        return ai.uid
    } catch (e: PackageManager.NameNotFoundException) {
        e.printStackTrace()
    }
    return -1
}

上面的代码是获取了所有手机里面的应用,然后在分别获取了他们的总大小,当然你要获取某个应用的总大小也是可以的,代码如下:

fun getAppTotalsizeO(path: File){
    try {
         val storageStatsManager: StorageStatsManager = context.getSystemService(Context.STORAGE_STATS_SERVICE)
as StorageStatsManager
        val storageManager: StorageManager = context.getSystemService(Context.STORAGE_SERVICE) as StorageManager
        val uuid: UUID = storageManager.getUuidForPath(path)
        //通过包名获取uid
        val uid = getUid(context, it.packageName)
        val storageStats = storageStatsManager.queryStatsForUid(uuid, uid)
        //获取到App的总大小
        appSizeL = storageStats.appBytes + storageStats.cacheBytes + storageStats.dataBytes               
    } catch (e: IOException) {
        e.printStackTrace()
    }
}

上面的方法,可能有同学不理解path需要传入什么?其实path就是用户数据目录,如果此路径为设备的默认内部存储getDataDirectory(),则storageManager.getUuidForPath(path)返回的值将为StorageManager.UUID_DEFAULT,通过上面的方法我们就可以获取某个应用的总大小了!以上就是Android 8.0以后获取应用总大小的方法,如果还想了解StorageManager 类更多的接口,可以去看官网接口说明文档

总结

通过以上的方法结合使用,我们就可以获取获取到不同版本手机的应用的总大小了,要适配不同版本的机型,就要结合不同版本的特性,合理的使用对应的方法!

  • 2
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 8
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 8
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值