在阅读阮一峰的《科技爱好者周刊(第165期)》时,看到了一个关于将字节数转换为人类可读的格式的文章,位置在:
文章->6.复制次数最多的 StackOverflow 代码片段,这段代码的作者多年后写了这篇文章,分析自己原始代码里面的漏洞。
文章详细描述了作者看到问题,写出了原始的解决方案代码的过程,分析了原始代码里的漏洞,并在文末提供了修复bug后的版本,以及可用于生产环境的优化后版本。
stackoverflow连接:
目前作者提供的可用于生产环境的代码已经更新到stackoverflow中,可以直接在stackoverflow中复制这段代码。
作者提供的原始版本,只有在SI模式(1K=1000)下才会有问题,因此如果我们的使用场景是输出文件大小等二进制Binary模式(1 K = 1024)的话,原始版本也是可以使用的。
public static String humanReadableByteCountSI(long bytes) {
if (-1000 < bytes && bytes < 1000) {
return bytes + " B";
}
CharacterIterator ci = new StringCharacterIterator("kMGTPE");
while (bytes <= -999_950 || bytes >= 999_950) {
bytes /= 1000;
ci.next();
}
return String.format("%.1f %cB", bytes / 1000.0, ci.current());
}
Binary (1 Ki = 1,024)
public static String humanReadableByteCountBin(long bytes) {
long absB = bytes == Long.MIN_VALUE ? Long.MAX_VALUE : Math.abs(bytes);
if (absB < 1024) {
return bytes + " B";
}
long value = absB;
CharacterIterator ci = new StringCharacterIterator("KMGTPE");
for (int i = 40; i >= 0 && absB > 0xfffccccccccccccL >> i; i -= 10) {
value >>= 10;
ci.next();
}
value *= Long.signum(bytes);
return String.format("%.1f %ciB", value / 1024.0, ci.current());
}
输出示例:



