目录
- 前言
- 获取存储容量
前言
原生系统设置里的存储容量到底是怎么计算的,跟踪源码,涉及到VolumeInfo、StorageManagerVolumeProvider、PrivateStorageInfo、StorageStatsManager......
等等,java上层没有办法使用简单的api获取到吗?搞了半天,总算获取到了,话不多说,直接上代码。
获取存储容量
/*** 内部总存储空间*/public static long getStorageTotal(Context context) {StorageStatsManager storageStatsManager = (StorageStatsManager) context.getSystemService(STORAGE_STATS_SERVICE);try {return storageStatsManager.getTotalBytes(StorageManager.UUID_DEFAULT);} catch (IOException e) {e.printStackTrace();}return 0;}/*** 内部可用空间*/public static long getStorageFree(Context context) {StorageStatsManager storageStatsManager = (StorageStatsManager) context.getSystemService(STORAGE_STATS_SERVICE);try {return storageStatsManager.getFreeBytes(StorageManager.UUID_DEFAULT);} catch (IOException e) {e.printStackTrace();}return 0;}/*** SD卡总空间*/public static long getSdCardTotal(Context context) {// 获取所有可用的存储卷File[] externalFiles = context.getExternalFilesDirs(null);for (File file : externalFiles) {if (Environment.isExternalStorageRemovable(file) && !file.getAbsolutePath().contains("emulated")) {StatFs stat = new StatFs(file.getPath());long blockSize = stat.getBlockSizeLong();long totalBlocks = stat.getBlockCountLong();return totalBlocks * blockSize;}}return 0;}/*** SD卡可用空间*/public static long getSdCardFree(Context context) {// 获取所有可用的存储卷File[] externalFiles = context.getExternalFilesDirs(null);for (File file : externalFiles) {if (Environment.isExternalStorageRemovable(file) && !file.getAbsolutePath().contains("emulated")) {StatFs stat = new StatFs(file.getPath());long blockSize = stat.getBlockSizeLong();long availableBlocks = stat.getAvailableBlocksLong();return availableBlocks * blockSize;}}return 0;}// 格式化文件大小为人类可读的格式private static String formatFileSize(long size) {String[] units = new String[]{"B", "KB", "MB", "GB", "TB"};int unitIndex = 0;double fileSize = size;while (fileSize > 1000 && unitIndex < units.length - 1) {fileSize = fileSize / 1000;unitIndex++;}return String.format("%.2f %s", fileSize, units[unitIndex]);}
打印下log看一下:
细心的小伙伴肯定发现单位换算是1000而不是1024了,为什么呢?
其实跟我们电脑上的硬盘存储一样,我们买硬盘的时候会发现1T的容量实际只有930G左右,厂商会解释说那是因为电脑系统采用的1024进制造成容量识别误差,因为厂商们生产的时候都是按照1T=1000G…1KB=1000B的进制的,现在手机厂商也学了这一套,标出64G,128G的容量也是按照1000进制的。 奸商啊,要是1024岂不是能放更多东西,哼,
UUID是什么?
UUID_DEFAULT 是一个常量,表示默认存储的 UUID。在这里,它代表默认存储卷的 UUID,通常指的是内部存储或者主要的外部存储(比如 SD 卡)。
在 Android 设备上,不同的存储卷可以具有不同的标识符(UUID),用于唯一标识每个存储卷。StorageManager.UUID_DEFAULT
是指向默认存储卷的 UUID,因此在获取存储信息时,使用这个 UUID 可以获得默认存储卷的存储空间信息。
更进一步的了解,参考谷歌官方:
UUID
StatFs
另外一点就是StatFs,StatFs 类是 Android 提供的用于获取文件系统信息的类,通常使用 StatFs 类来获取存储卷(如内部存储、外部 SD 卡)的文件系统信息,在应用程序中对存储空间进行监测和管理。通过 StatFs 可以获取存储卷的总大小、可用大小、文件块的大小等信息,我们这里就是使用不同文件系统的块数*每块的字节数
进而获取总大小的。