Ik moet een bestandsgrootte weergeven als een tekenreeksmet zinnige eenheden.
Bijvoorbeeld
1L ==> "1 B";
1024L ==> "1 KB";
2537253L ==> "2.3 MB"
enz.
Ik vond dit eerdere antwoord, dat ik niet bevredigend vond .
Ik heb mijn eigen oplossing bedacht die vergelijkbare tekortkomingen heeft:
private static final long K = 1024;
private static final long M = K * K;
private static final long G = M * K;
private static final long T = G * K;
public static String convertToStringRepresentation(final long value){
final long[] dividers = new long[] { T, G, M, K, 1 };
final String[] units = new String[] { "TB", "GB", "MB", "KB", "B" };
if(value < 1)
throw new IllegalArgumentException("Invalid file size: " + value);
String result = null;
for(int i = 0; i < dividers.length; i++){
final long divider = dividers[i];
if(value >= divider){
result = format(value, divider, units[i]);
break;
}
}
return result;
}
private static String format(final long value,
final long divider,
final String unit){
final double result =
divider > 1 ? (double) value / (double) divider : (double) value;
return String.format("%.1f %s", Double.valueOf(result), unit);
}
Het grootste probleem is mijn beperkte kennis van Decimalformat en/of String.format. Ik wil dat 1024L, 1025L, enz. worden toegewezen aan 1 KB
in plaats van 1.0 KB
.
Dus twee mogelijkheden:
- Ik heb liever een goede kant-en-klare oplossing in een openbare bibliotheek zoals Apache Commonsof Google Guava.
- Als dat niet het geval is, hoe kan ik dan het ‘.0’-gedeelte verwijderen (zonder toevlucht te nemen tot stringvervanging en regex, dat kan ik zelf doen)?
Antwoord 1, autoriteit 100%
public static String readableFileSize(long size) {
if(size <= 0) return "0";
final String[] units = new String[] { "B", "kB", "MB", "GB", "TB" };
int digitGroups = (int) (Math.log10(size)/Math.log10(1024));
return new DecimalFormat("#,##0.#").format(size/Math.pow(1024, digitGroups)) + " " + units[digitGroups];
}
Dit werkt tot 1000 TB…. en het programma is kort!
Antwoord 2, autoriteit 2%
Je hebt waarschijnlijk meer geluk met java.text.DecimalFormat
. Deze code zou het waarschijnlijk moeten doen (alleen maar vleugelen…)
new DecimalFormat("#,##0.#").format(value) + " " + unit
Antwoord 3
Het is verrassend voor mij, maar een loop-gebaseerd algoritme is ongeveer 10% sneller.
public static String toNumInUnits(long bytes) {
int u = 0;
for ( ; bytes > 1024*1024; bytes >>= 10) {
u++;
}
if (bytes > 1024)
u++;
return String.format("%.1f %cB", bytes/1024f, " kMGTPE".charAt(u));
}