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
类更多的接口,可以去看官网接口说明文档
总结
通过以上的方法结合使用,我们就可以获取获取到不同版本手机的应用的总大小了,要适配不同版本的机型,就要结合不同版本的特性,合理的使用对应的方法!