วิธีการแปลงขนาดไบต์เป็นรูปแบบที่มนุษย์อ่านได้ใน Java?


556

วิธีการแปลงขนาดไบต์เป็นรูปแบบที่มนุษย์อ่านได้ใน Java? เช่นเดียวกับ 1024 ควรกลายเป็น "1 Kb" และ 1024 * 1024 ควรกลายเป็น "1 Mb"

ฉันเบื่อเขียนวิธีอรรถประโยชน์นี้สำหรับแต่ละโครงการ มีวิธีการคงที่ในApache Commonsนี้หรือไม่


32
หากคุณใช้หน่วยมาตรฐาน 1024 ควรเป็น "1KiB" และ 1024 * 1024 ควรกลายเป็น "1MiB" en.wikipedia.org/wiki/Binary_prefix
Pascal Cuoq

@Pascal: ควรมีฟังก์ชั่นหลายอย่างหรือตัวเลือกเพื่อระบุฐานและหน่วย
Aaron Digulla


3
@Pascal Cuoq: ขอบคุณสำหรับการอ้างอิง ฉันไม่รู้จนกระทั่งฉันอ่านว่าที่นี่ในสหภาพยุโรปเราจำเป็นต้องใช้คำนำหน้าถูกต้องตามกฎหมาย
JeremyP

2
@DerMike คุณพูดถึงว่า "จนกว่าจะมีห้องสมุดดังกล่าวอยู่" ตอนนี้กลายเป็นจริง :-) stackoverflow.com/questions/3758606/…
Christian Esken

คำตอบ:


1310

ความจริงแล้วสนุก:ข้อมูลโค้ดต้นฉบับที่โพสต์ที่นี่เป็นข้อมูลโค้ด Java ที่ถูกคัดลอกมากที่สุดของเวลาทั้งหมดใน Stack Overflow และมีข้อบกพร่อง มันได้รับการแก้ไข แต่มันก็ยุ่ง

เรื่องเต็มในบทความนี้: ตัวอย่างของ StackOverflow ที่ถูกคัดลอกมากที่สุดตลอดเวลามีข้อบกพร่อง!

แหล่งที่มา: การจัดรูปแบบขนาดไบต์เป็นรูปแบบที่มนุษย์สามารถอ่านได้ Programming.Guide

SI (1 k = 1,000)

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());
}

ไบนารี (1 K = 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());
}

ตัวอย่างผลลัพธ์:

                              SI     BINARY

                   0:        0 B        0 B
                  27:       27 B       27 B
                 999:      999 B      999 B
                1000:     1.0 kB     1000 B
                1023:     1.0 kB     1023 B
                1024:     1.0 kB    1.0 KiB
                1728:     1.7 kB    1.7 KiB
              110592:   110.6 kB  108.0 KiB
             7077888:     7.1 MB    6.8 MiB
           452984832:   453.0 MB  432.0 MiB
         28991029248:    29.0 GB   27.0 GiB
       1855425871872:     1.9 TB    1.7 TiB
 9223372036854775807:     9.2 EB    8.0 EiB   (Long.MAX_VALUE)

12
ฉันชอบ 1.0 KB จากนั้นจะเห็นได้อย่างชัดเจนว่าตัวเลขที่สำคัญ ๆ (นี่ก็ดูเหมือนจะเป็นพฤติกรรมของตัวอย่างเช่นduคำสั่งใน Linux.)
aioobe

19
ฉันคิดว่าทุกคนควรทราบว่าในโครงการลูกค้าของคุณต้องการดูค่าในฐาน 2 (เบี่ยงเบนจาก 1024) แต่ด้วยคำนำหน้าทั่วไป ไม่ใช่ KiB, MiB, GiB เป็นต้นใช้ KB, MB, GB, TB สำหรับมัน
Borys

27
@Borys การใช้ "KB" เพื่อหมายถึง "1024 bytes" ผิด อย่าทำอย่างนั้น
endolith

8
ผู้อ่านจะได้เรียนรู้มัน ดีกว่าสิ่งที่พวกเขาไม่คุ้นเคยและสามารถเรียนรู้ได้ดีกว่ามีบางอย่างผิดปกติ การเขียน KB ผู้ใช้ที่คุ้นเคยจะคาดว่า 1,000 และผู้ใช้ที่ไม่คุ้นเคยจะคาดหวัง 1024
kap

16
คำตอบเขียนใหม่ทั้งหมด ความคิดเห็นด้านบนจำนวนมากล้าสมัย
aioobe

305

FileUtils.byteCountToDisplaySize(long size)org.apache.commons.ioจะทำงานถ้าโครงการของคุณจะขึ้นอยู่กับ

JavaDoc สำหรับวิธีนี้


18
ฉันมีคอมมอนส์ -oo แล้วในโครงการของฉัน แต่จบลงด้วยการใช้รหัสของ aioobe เนื่องจากพฤติกรรมการปัดเศษ (ดูลิงค์สำหรับ JavaDoc)
Iravanchi

3
มียูทิลิตีเพื่อใช้ในการดำเนินการย้อนกลับหรือไม่ รับจำนวนไบต์จากจำนวนไบต์ที่มนุษย์อ่านได้หรือไม่
arunmoezhi

6
น่าเสียดายที่ฟังก์ชั่นนี้ไม่ได้ตระหนักถึงตำแหน่งที่ตั้ง ในภาษาฝรั่งเศสตัวอย่างเช่นพวกเขาจะเรียก bytes "octets" เสมอดังนั้นหากคุณจะแสดงไฟล์ 100 KB ไปยังผู้ใช้ชาวฝรั่งเศสฉลากที่ถูกต้องจะเป็น 100 Ko
Tacroy

@Tacroy คุณสามารถรับเอาต์พุต octets ด้วย UnitFormatter ในไลบรารี triava คุณสามารถผ่านหน่วยใด ๆ เป็นไบต์วัตต์หรือ octets ตัวอย่างแก้ไขเล็กน้อยจากตัวอย่างในgithub.com/trivago/triava : UnitFormatter.formatAsUnit (1126, UnitSystem.SI, "o"); // = "1.13 ko" ตัวอย่างเพิ่มเติมใน: stackoverflow.com/questions/3758606/…
คริสเตียน Esken

5
รอบนี้เป็น gb ที่ใกล้ที่สุดเมื่อ> 1 gb ซึ่งหมายความว่าความแม่นยำที่คุณได้รับนั้นแตกต่างกันไป
tksfz

180

ใช้ Android builtin Class

สำหรับ Android มีระดับฟอร์แมต เพียงหนึ่งบรรทัดของรหัสและคุณเสร็จแล้ว

android.text.format.Formatter.formatShortFileSize(activityContext, bytes);

มันเป็นเหมือนformatFileSize()แต่พยายามที่จะสร้างตัวเลขที่สั้นกว่า (แสดงทศนิยมน้อยลง)

android.text.format.Formatter.formatFileSize(activityContext, bytes);

จัดรูปแบบขนาดเนื้อหาให้อยู่ในรูปแบบของไบต์, กิโลไบต์, เมกะไบต์ ฯลฯ


12
ควรเป็นคำตอบที่ดีที่สุดสำหรับ ANDROID อย่างแน่นอน ไม่จำเป็นต้องใช้ห้องสมุดเพิ่มเติม 1
หิวโหย

11
Contextฉันเกลียดความจริงที่คุณต้องผ่านใน
Jared Burrows

4
ควรเป็นคำตอบที่ดีที่สุดสำหรับ ANDROID อย่างแน่นอน
shridutt kothari

5
คุณส่งผ่านบริบทเพื่อแปลเป็นภาษาปัจจุบันของผู้ใช้ มิฉะนั้นมันจะไม่เป็นฟังก์ชั่นที่มีประโยชน์มาก
phreakhead

7
ฉันใช้คำตอบที่ยอมรับก่อนที่ฉันจะรู้สิ่งนี้ สิ่งที่จะสังเกตเห็นได้ใน Build.VERSION_CODES.N และก่อนหน้านั้นใช้กำลังของ 1024 แทนด้วย KB = 1024 ไบต์, MB = 1,048,576 ไบต์เป็นต้น ณ O คำนำหน้าจะใช้ในความหมายมาตรฐานในระบบ SI ดังนั้น kB = 1,000 ไบต์, MB = 1,000,000 ไบต์ ฯลฯ
HendraWD

57

เราสามารถหลีกเลี่ยงการใช้ความช้าMath.pow()และMath.log()วิธีการโดยไม่ต้องเสียสละความเรียบง่ายอย่างสมบูรณ์เนื่องจากปัจจัยระหว่างหน่วย (เช่น B, KB, MB ฯลฯ ) คือ 1024 ซึ่งเท่ากับ 2 ^ 10 Longชั้นจะมีประโยชน์numberOfLeadingZeros()วิธีการที่เราสามารถใช้ในการบอกว่าหน่วยมูลค่าขนาดตกอยู่ใน

จุดสำคัญ:หน่วยขนาดมีระยะทาง 10 บิต (1024 = 2 ^ 10) หมายถึงตำแหน่งสูงสุด 1 บิต - หรืออีกนัยหนึ่งคือจำนวนศูนย์นำหน้า - ต่างกัน 10 (ไบต์ = KB * 1024, KB = MB) * 1024 และอื่น ๆ )

ความสัมพันธ์ระหว่างจำนวนศูนย์นำและหน่วยขนาด:

# of leading 0's   Size unit
-------------------------------
>53                B (Bytes)
>43                KB
>33                MB
>23                GB
>13                TB
>3                 PB
<=2                EB

รหัสสุดท้าย:

public static String formatSize(long v) {
    if (v < 1024) return v + " B";
    int z = (63 - Long.numberOfLeadingZeros(v)) / 10;
    return String.format("%.1f %sB", (double)v / (1L << (z*10)), " KMGTPE".charAt(z));
}

24

ฉันถามคำถามเดียวกันเมื่อเร็ว ๆ นี้:

จัดรูปแบบขนาดไฟล์เป็น MB, GB เป็นต้น

ในขณะที่ไม่มีคำตอบนอกกรอบฉันสามารถอยู่กับวิธีแก้ปัญหา:

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 new DecimalFormat("#,##0.#").format(result) + " " + unit;
}

รหัสทดสอบ:

public static void main(final String[] args){
    final long[] l = new long[] { 1l, 4343l, 43434334l, 3563543743l };
    for(final long ll : l){
        System.out.println(convertToStringRepresentation(ll));
    }
}

เอาท์พุท (ในสถานที่เยอรมันของฉัน):

1 B
4,2 KB
41,4 MB
3,3 GB

แก้ไข: ฉันได้เปิดออกขอฟังก์ชั่นนี้สำหรับ Google ฝรั่ง บางทีบางคนก็อยากจะสนับสนุน


2
เหตุใดขนาดไฟล์จึงไม่ถูกต้อง
aioobe

@aioobe มันเป็นกรณีการใช้งานของฉัน (แสดงขนาดของไฟล์ที่อัปโหลด) แต่เนื้อหานั้นไม่ใช่สากล
Sean Patrick Floyd

หากคุณเปลี่ยนบรรทัดสุดท้ายเพื่อส่งคืน NumberFormat.getFormat ("#, ## 0. #"). format (result) + "" + หน่วย; มันใช้งานได้ใน GWT ด้วย! ขอบคุณสำหรับสิ่งนี้มันยังไม่ได้อยู่ใน Guava
ทอม

9

นี้เป็นรุ่นที่ปรับเปลี่ยนจากคำตอบของ aioobe

การเปลี่ยนแปลง:

  • Localeพารามิเตอร์เนื่องจากบางภาษาใช้.และอื่น ๆ,เป็นจุดทศนิยม
  • รหัสที่มนุษย์อ่านได้

private static final String[] SI_UNITS = { "B", "kB", "MB", "GB", "TB", "PB", "EB" };
private static final String[] BINARY_UNITS = { "B", "KiB", "MiB", "GiB", "TiB", "PiB", "EiB" };

public static String humanReadableByteCount(final long bytes, final boolean useSIUnits, final Locale locale)
{
    final String[] units = useSIUnits ? SI_UNITS : BINARY_UNITS;
    final int base = useSIUnits ? 1000 : 1024;

    // When using the smallest unit no decimal point is needed, because it's the exact number.
    if (bytes < base) {
        return bytes + " " + units[0];
    }

    final int exponent = (int) (Math.log(bytes) / Math.log(base));
    final String unit = units[exponent];
    return String.format(locale, "%.1f %s", bytes / Math.pow(base, exponent), unit);
}

มันเป็นผลลัพธ์ที่ผสมกันเล็กน้อยในการส่งต่อพารามิเตอร์ Locale สำหรับสัญลักษณ์ตัวแยกเท่านั้น แต่จากนั้นไม่ต้อง จำกัด หน่วยให้กับภาษาที่ใช้สัญลักษณ์อื่นสำหรับ Bytes เช่นภาษาฝรั่งเศส
Nzall

@Nzall คุณหมายถึง octet หรือเปล่า Wikipedia ระบุว่าไม่เป็นเรื่องปกติอีกต่อไป อื่นคุณมีการอ้างอิงหรือไม่?
Christian Strempfer

7

หากคุณใช้ Android ของคุณก็สามารถใช้android.text.format.Formatter.formatFileSize ()

ทางเลือกต่อไปนี้เป็นวิธีการแก้ไขตามโพสต์ยอดนิยมนี้ :

  /**
   * formats the bytes to a human readable format
   *
   * @param si true if each kilo==1000, false if kilo==1024
   */
  @SuppressLint("DefaultLocale")
  public static String humanReadableByteCount(final long bytes,final boolean si)
    {
    final int unit=si ? 1000 : 1024;
    if(bytes<unit)
      return bytes+" B";
    double result=bytes;
    final String unitsToUse=(si ? "k" : "K")+"MGTPE";
    int i=0;
    final int unitsCount=unitsToUse.length();
    while(true)
      {
      result/=unit;
      if(result<unit)
        break;
      // check if we can go further:
      if(i==unitsCount-1)
        break;
      ++i;
      }
    final StringBuilder sb=new StringBuilder(9);
    sb.append(String.format("%.1f ",result));
    sb.append(unitsToUse.charAt(i));
    if(si)
      sb.append('B');
    else sb.append('i').append('B');
    final String resultStr=sb.toString();
    return resultStr;
    }

หรือใน Kotlin:

/**
 * formats the bytes to a human readable format
 *
 * @param si true if each kilo==1000, false if kilo==1024
 */
@SuppressLint("DefaultLocale")
fun humanReadableByteCount(bytes: Long, si: Boolean): String? {
    val unit = if (si) 1000.0 else 1024.0
    if (bytes < unit)
        return "$bytes B"
    var result = bytes.toDouble()
    val unitsToUse = (if (si) "k" else "K") + "MGTPE"
    var i = 0
    val unitsCount = unitsToUse.length
    while (true) {
        result /= unit
        if (result < unit || i == unitsCount - 1)
            break
        ++i
    }
    return with(StringBuilder(9)) {
        append(String.format("%.1f ", result))
        append(unitsToUse[i])
        if (si) append('B') else append("iB")
    }.toString()
}

คุณดูเหมือนจะมีข้อผิดพลาดแบบออฟไลน์ใน for-loop ของคุณ ผมคิดว่ามันควรจะเป็นและไม่ได้unitsCount unitsCount-1
aioobe

@aioobe แต่นี่หมายถึงการวนซ้ำสามารถหยุดเมื่อ i == unitsCount ซึ่งหมายถึง i == 6 ซึ่งหมายถึง "charAt" จะล้มเหลว ...
นักพัฒนา Android

if(result<unit) break;จะเตะก่อนหน้านั้น ไม่ต้องห่วง. (ถ้าคุณทดสอบคุณจะสังเกตเห็นว่าคุณสามารถข้ามเงื่อนไขในวงอย่างสิ้นเชิง.)
aioobe

@aioobe ถูกต้องนั่นเป็นเพราะข้อสันนิษฐาน (ซึ่งถูกต้อง) ที่ฉันจัดการกับตัวแปรชนิด "ยาว" นอกจากนี้ยังขึ้นอยู่กับสมมติฐานว่าหน่วยจะเป็นอย่างน้อยสิ่งที่ฉันเขียน หากคุณใช้หน่วยน้อยลงมันจะสร้างผลลัพธ์แปลก ๆ (จะต้องการค่าที่น้อยกว่า 1 มากกว่าค่าที่มากกว่า 1,000)
นักพัฒนา android

@aioobe ถูกต้อง ฉันจะซ่อมมัน. BTW อัลกอริทึมของคุณยังสามารถให้ผลลัพธ์ที่แปลก ลองให้ "999999 จริง" เป็นอาร์กิวเมนต์ มันจะแสดง "1,000.0 kB" ดังนั้นมันจะถูกปัดเศษ แต่เมื่อผู้คนเห็นมันพวกเขาสงสัยว่า: ทำไมมันไม่แสดง 1MB เป็น 1000KB = 1MB ... คุณคิดว่าเรื่องนี้ควรได้รับการจัดการอย่างไร มันเป็นเพราะ String.format แต่ฉันไม่แน่ใจว่าควรแก้ไขอย่างไร
นักพัฒนา android

6

private static final String[] Q = new String[]{"", "K", "M", "G", "T", "P", "E"};

public String getAsString(long bytes)
{
    for (int i = 6; i > 0; i--)
    {
        double step = Math.pow(1024, i);
        if (bytes > step) return String.format("%3.1f %s", bytes / step, Q[i]);
    }
    return Long.toString(bytes);
}

6
  private String bytesIntoHumanReadable(long bytes) {
        long kilobyte = 1024;
        long megabyte = kilobyte * 1024;
        long gigabyte = megabyte * 1024;
        long terabyte = gigabyte * 1024;

        if ((bytes >= 0) && (bytes < kilobyte)) {
            return bytes + " B";

        } else if ((bytes >= kilobyte) && (bytes < megabyte)) {
            return (bytes / kilobyte) + " KB";

        } else if ((bytes >= megabyte) && (bytes < gigabyte)) {
            return (bytes / megabyte) + " MB";

        } else if ((bytes >= gigabyte) && (bytes < terabyte)) {
            return (bytes / gigabyte) + " GB";

        } else if (bytes >= terabyte) {
            return (bytes / terabyte) + " TB";

        } else {
            return bytes + " Bytes";
        }
    }

ฉันชอบสิ่งนี้เพราะมันง่ายที่จะติดตามและเข้าใจง่าย
Joshua Pinter

6

หน่วยไบต์ช่วยให้คุณทำเช่นนี้:

long input1 = 1024;
long input2 = 1024 * 1024;

Assert.assertEquals("1 KiB", BinaryByteUnit.format(input1));
Assert.assertEquals("1 MiB", BinaryByteUnit.format(input2));

Assert.assertEquals("1.024 KB", DecimalByteUnit.format(input1, "#.0"));
Assert.assertEquals("1.049 MB", DecimalByteUnit.format(input2, "#.000"));

NumberFormat format = new DecimalFormat("#.#");
Assert.assertEquals("1 KiB", BinaryByteUnit.format(input1, format));
Assert.assertEquals("1 MiB", BinaryByteUnit.format(input2, format));

ฉันได้เขียนไลบรารีอื่นที่เรียกว่าหน่วยจัดเก็บข้อมูลที่ให้คุณทำสิ่งนี้:

String formattedUnit1 = StorageUnits.formatAsCommonUnit(input1, "#");
String formattedUnit2 = StorageUnits.formatAsCommonUnit(input2, "#");
String formattedUnit3 = StorageUnits.formatAsBinaryUnit(input1);
String formattedUnit4 = StorageUnits.formatAsBinaryUnit(input2);
String formattedUnit5 = StorageUnits.formatAsDecimalUnit(input1, "#.00", Locale.GERMAN);
String formattedUnit6 = StorageUnits.formatAsDecimalUnit(input2, "#.00", Locale.GERMAN);
String formattedUnit7 = StorageUnits.formatAsBinaryUnit(input1, format);
String formattedUnit8 = StorageUnits.formatAsBinaryUnit(input2, format);

Assert.assertEquals("1 kB", formattedUnit1);
Assert.assertEquals("1 MB", formattedUnit2);
Assert.assertEquals("1.00 KiB", formattedUnit3);
Assert.assertEquals("1.00 MiB", formattedUnit4);
Assert.assertEquals("1,02 kB", formattedUnit5);
Assert.assertEquals("1,05 MB", formattedUnit6);
Assert.assertEquals("1 KiB", formattedUnit7);
Assert.assertEquals("1 MiB", formattedUnit8);

ในกรณีที่คุณต้องการบังคับให้ทำเช่นนี้:

String formattedUnit9 = StorageUnits.formatAsKibibyte(input2);
String formattedUnit10 = StorageUnits.formatAsCommonMegabyte(input2);

Assert.assertEquals("1024.00 KiB", formattedUnit9);
Assert.assertEquals("1.00 MB", formattedUnit10);

5
    public static String floatForm (double d)
    {
       return new DecimalFormat("#.##").format(d);
    }


    public static String bytesToHuman (long size)
    {
        long Kb = 1  * 1024;
        long Mb = Kb * 1024;
        long Gb = Mb * 1024;
        long Tb = Gb * 1024;
        long Pb = Tb * 1024;
        long Eb = Pb * 1024;

        if (size <  Kb)                 return floatForm(        size     ) + " byte";
        if (size >= Kb && size < Mb)    return floatForm((double)size / Kb) + " Kb";
        if (size >= Mb && size < Gb)    return floatForm((double)size / Mb) + " Mb";
        if (size >= Gb && size < Tb)    return floatForm((double)size / Gb) + " Gb";
        if (size >= Tb && size < Pb)    return floatForm((double)size / Tb) + " Tb";
        if (size >= Pb && size < Eb)    return floatForm((double)size / Pb) + " Pb";
        if (size >= Eb)                 return floatForm((double)size / Eb) + " Eb";

        return "???";
    }

3

ขณะนี้มีหนึ่งไลบรารีที่มีการจัดรูปแบบหน่วย ฉันเพิ่มมันลงในไลบรารีtriavaเนื่องจากดูเหมือนว่ามีเพียงไลบรารีเดียวที่มีอยู่สำหรับ Android

สามารถจัดรูปแบบตัวเลขด้วยความแม่นยำโดยพลการใน 3 ระบบที่แตกต่างกัน (SI, IEC, JEDEC) และตัวเลือกเอาต์พุตต่างๆ นี่คือตัวอย่างโค้ดจากการทดสอบหน่วย Triava :

UnitFormatter.formatAsUnit(1126, UnitSystem.SI, "B");
// = "1.13kB"
UnitFormatter.formatAsUnit(2094, UnitSystem.IEC, "B");
// = "2.04KiB"

การพิมพ์ค่า kilo ที่แน่นอนค่า mega (ที่นี่พร้อม W = วัตต์):

UnitFormatter.formatAsUnits(12_000_678, UnitSystem.SI, "W", ", ");
// = "12MW, 678W"

คุณสามารถส่ง DecimalFormat เพื่อปรับแต่งเอาต์พุต:

UnitFormatter.formatAsUnit(2085, UnitSystem.IEC, "B", new DecimalFormat("0.0000"));
// = "2.0361KiB"

สำหรับการดำเนินการโดยพลการเกี่ยวกับค่าเป็นกิโลหรือ mega คุณสามารถแบ่งออกเป็นองค์ประกอบ:

UnitComponent uc = new  UnitComponent(123_345_567_789L, UnitSystem.SI);
int kilos = uc.kilo(); // 567
int gigas = uc.giga(); // 123

2

ฉันรู้ว่ามันสายเกินไปที่จะอัพเดทโพสต์นี้! แต่ฉันสนุกกับเรื่องนี้:

สร้างส่วนต่อประสาน:

public interface IUnits {
     public String format(long size, String pattern);
     public long getUnitSize();
}

สร้างคลาส StorageUnits:

import java.text.DecimalFormat;

public class StorageUnits {
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;

enum Unit implements IUnits {
    TERA_BYTE {
        @Override
        public String format(long size, String pattern) {
            return format(size, getUnitSize(), "TB", pattern);
        }
        @Override
        public long getUnitSize() {
            return T;
        }
        @Override
        public String toString() {
            return "Terabytes";
        }
    },
    GIGA_BYTE {
        @Override
        public String format(long size, String pattern) {
            return format(size, getUnitSize(), "GB", pattern);
        }
        @Override
        public long getUnitSize() {
            return G;
        }
        @Override
        public String toString() {
            return "Gigabytes";
        }
    },
    MEGA_BYTE {
        @Override
        public String format(long size, String pattern) {
            return format(size, getUnitSize(), "MB", pattern);
        }
        @Override
        public long getUnitSize() {
            return M;
        }
        @Override
        public String toString() {
            return "Megabytes";
        }
    },
    KILO_BYTE {
        @Override
        public String format(long size, String pattern) {
            return format(size, getUnitSize(), "kB", pattern);
        }
        @Override
        public long getUnitSize() {
            return K;
        }
        @Override
        public String toString() {
            return "Kilobytes";
        }

    };
    String format(long size, long base, String unit, String pattern) {
        return new DecimalFormat(pattern).format(
                Long.valueOf(size).doubleValue() / Long.valueOf(base).doubleValue()
        ) + unit;
    }
}

public static String format(long size, String pattern) {
    for(Unit unit : Unit.values()) {
        if(size >= unit.getUnitSize()) {
            return unit.format(size, pattern);
        }
    }
    return ("???(" + size + ")???");
}

public static String format(long size) {
    return format(size, "#,##0.#");
}
}

เรียกมันว่า:

class Main {
    public static void main(String... args) {
         System.out.println(StorageUnits.format(21885));
         System.out.println(StorageUnits.format(2188121545L));
    }
}

เอาท์พุท:

21.4kB
2GB

2

ในกรณีที่ไม่มีโอกาสมันช่วยประหยัดเวลาของใครบางคนหรืออาจจะเพื่อความสนุกนี่เป็นเวอร์ชั่น Go เพื่อความง่ายฉันได้รวมเอาท์พุทไบนารีเท่านั้น

func sizeOf(bytes int64) string {
    const unit = 1024
    if bytes < unit {
        return fmt.Sprintf("%d B", bytes)
    }

    fb := float64(bytes)
    exp := int(math.Log(fb) / math.Log(unit))
    pre := "KMGTPE"[exp-1]
    div := math.Pow(unit, float64(exp))
    return fmt.Sprintf("%.1f %ciB", fb / div, pre)
}

1
String[] fileSizeUnits = {"bytes", "KB", "MB", "GB", "TB", "PB", "EB", "ZB", "YB"};
public String calculateProperFileSize(double bytes){
    String sizeToReturn = "";
    int index = 0;
    for(index = 0; index < fileSizeUnits.length; index++){
        if(bytes < 1024){
            break;
        }
        bytes = bytes / 1024;
    }

เพียงแค่เพิ่มหน่วยไฟล์เพิ่มเติม (หากมีหายไป) และคุณจะเห็นขนาดหน่วยเกินหน่วยนั้น (หากไฟล์ของคุณมีความยาวมากขนาดนั้น) System.out.println ("ขนาดไฟล์ในรูปแบบที่เหมาะสม:" + bytes + "" + fileSizeUnits [ดัชนี]); sizeToReturn = String.valueOf (ไบต์) + "" + fileSizeUnits [ดัชนี]; ผลตอบแทนขนาดกลับมา; }


1

นี่คือ C # .net เทียบเท่าสำหรับ Java คำตอบที่ถูกต้องฉันทามติด้านบน (มีอีกด้านล่างซึ่งมีรหัสสั้นกว่า)

    public static String BytesNumberToHumanReadableString(long bytes, bool SI1000orBinary1024)
    {

        int unit = SI1000orBinary1024 ? 1000 : 1024;
        if (bytes < unit) return bytes + " B";
        int exp = (int)(Math.Log(bytes) / Math.Log(unit));
        String pre = (SI1000orBinary1024 ? "kMGTPE" : "KMGTPE")[(exp - 1)] + (SI1000orBinary1024 ? "" : "i");
        return String.Format("{0:F1} {1}B", bytes / Math.Pow(unit, exp), pre);
    }

เทคนิคการพูดถ้าเรายึดติดกับหน่วย SI รูทีนนี้ใช้ได้กับการใช้ตัวเลขเป็นประจำ มีคำตอบที่ดีจากผู้เชี่ยวชาญมากมาย สมมติว่าคุณกำลังทำฐานข้อมูลตัวเลขอยู่บน gridviews มันคุ้มค่าที่จะตรวจสอบประสิทธิภาพการทำงานที่ได้รับการปรับปรุงจากพวกเขา

PS: โพสต์เพราะคำถาม / คำตอบนี้ขึ้นอยู่บนการค้นหาของ Google ในขณะที่ฉันกำลังทำโครงการ C #




0
filename=filedilg.getSelectedFile().getAbsolutePath();
File file=new File(filename);

String disp=FileUtils.byteCountToDisplaySize(file.length());
System.out.println("THE FILE PATH IS "+file+"THIS File SIZE IS IN MB "+disp);

คำตอบนี้ในขณะที่ใช้งานได้เป็นส่วนเติมเต็มให้กับคำตอบก่อนหน้าในหัวข้อนี้โดย @ user601806: stackoverflow.com/a/4888400/3987745เพื่อให้คำตอบนี้ใช้งานได้คุณต้องใช้ Apache Commons IO ( commons.apache.org/proper/) Commons-io ) การพึ่งพา
Edward Quixote

0

คุณลองJSR 363 แล้วหรือยัง โมดูลส่วนขยายหน่วยเช่น Unicode CLDR (ในGitHub: uom-systems ) ทำทุกอย่างให้คุณ

คุณสามารถใช้MetricPrefixรวมอยู่ในทุกการใช้งานหรือBinaryPrefix(เทียบเคียงกับตัวอย่างข้างต้นบางส่วน) และถ้าคุณเช่นอาศัยและทำงานในอินเดียหรือประเทศใกล้เคียงIndianPrefix(เช่นในโมดูลทั่วไปของระบบ uom) ช่วยให้คุณสามารถใช้และจัดรูปแบบ "Crore" ไบต์ "หรือ" Lakh Bytes "เช่นกัน


0

บางทีคุณสามารถใช้รหัสนี้ (ใน C #):

        long Kb = 1024;
        long Mb = Kb * 1024;
        long Gb = Mb * 1024;
        long Tb = Gb * 1024;
        long Pb = Tb * 1024;
        long Eb = Pb * 1024;

        if (size < Kb) return size.ToString() + " byte";
        if (size < Mb) return (size / Kb).ToString("###.##") + " Kb.";
        if (size < Gb) return (size / Mb).ToString("###.##") + " Mb.";
        if (size < Tb) return (size / Gb).ToString("###.##") + " Gb.";
        if (size < Pb) return (size / Tb).ToString("###.##") + " Tb.";
        if (size < Eb) return (size / Pb).ToString("###.##") + " Pb.";
        if (size >= Eb) return (size / Eb).ToString("###.##") + " Eb.";

        return "invalid size";

0
public String humanReadable(long size) {
    long limit = 10 * 1024;
    long limit2 = limit * 2 - 1;
    String negative = "";
    if(size < 0) {
        negative = "-";
        size = Math.abs(size);
    }

    if(size < limit) {
        return String.format("%s%s bytes", negative, size);
    } else {
        size = Math.round((double) size / 1024);
        if (size < limit2) {
            return String.format("%s%s kB", negative, size);
        } else {
            size = Math.round((double)size / 1024);
            if (size < limit2) {
                return String.format("%s%s MB", negative, size);
            } else {
                size = Math.round((double)size / 1024);
                if (size < limit2) {
                    return String.format("%s%s GB", negative, size);
                } else {
                    size = Math.round((double)size / 1024);
                        return String.format("%s%s TB", negative, size);
                }
            }
        }
    }
}

0

ใช้ฟังก์ชั่นต่อไปนี้เพื่อรับข้อมูลที่แน่นอนสร้างโดยยึดATM_CashWithdrawlแนวคิดพื้นฐาน

getFullMemoryUnit(): Total: [123 MB], Max: [1 GB, 773 MB, 512 KB], Free: [120 MB, 409 KB, 304 Bytes]
public static String getFullMemoryUnit(long unit) {
    long BYTE = 1024, KB = BYTE, MB = KB * KB, GB = MB * KB, TB = GB * KB;
    long KILO_BYTE, MEGA_BYTE = 0, GIGA_BYTE = 0, TERA_BYTE = 0;
    unit = Math.abs(unit);
    StringBuffer buffer = new StringBuffer();
    if ( unit / TB > 0 ) {
        TERA_BYTE = (int) (unit / TB);
        buffer.append(TERA_BYTE+" TB");
        unit -= TERA_BYTE * TB;
    }
    if ( unit / GB > 0 ) {
        GIGA_BYTE = (int) (unit / GB);
        if (TERA_BYTE != 0) buffer.append(", ");
        buffer.append(GIGA_BYTE+" GB");
        unit %= GB;
    }
    if ( unit / MB > 0 ) {
        MEGA_BYTE = (int) (unit / MB);
        if (GIGA_BYTE != 0) buffer.append(", ");
        buffer.append(MEGA_BYTE+" MB");
        unit %= MB;
    }
    if ( unit / KB > 0 ) {
        KILO_BYTE = (int) (unit / KB);
        if (MEGA_BYTE != 0) buffer.append(", ");
        buffer.append(KILO_BYTE+" KB");
        unit %= KB;
    }
    if ( unit > 0 ) buffer.append(", "+unit+" Bytes");
    return buffer.toString();
}

ฉันเพิ่งแก้ไขรหัสของfacebookarchive-StringUtilsเพื่อให้ได้รูปแบบด้านล่าง รูปแบบเดียวกันคุณจะได้รับเมื่อคุณใช้apache.hadoop-StringUtils

getMemoryUnit(): Total: [123.0 MB], Max: [1.8 GB], Free: [120.4 MB]
public static String getMemoryUnit(long bytes) {
    DecimalFormat oneDecimal = new DecimalFormat("0.0");
    float BYTE = 1024.0f, KB = BYTE, MB = KB * KB, GB = MB * KB, TB = GB * KB;
    long absNumber = Math.abs(bytes);
    double result = bytes;
    String suffix = " Bytes";
    if (absNumber < MB) {
        result = bytes / KB;
        suffix = " KB";
    } else if (absNumber < GB) {
        result = bytes / MB;
        suffix = " MB";
    } else if (absNumber < TB) {
        result = bytes / GB;
        suffix = " GB";
    }
    return oneDecimal.format(result) + suffix;
}

ตัวอย่างการใช้วิธีการด้านบน:

public static void main(String[] args) {
    Runtime runtime = Runtime.getRuntime();
    int availableProcessors = runtime.availableProcessors();

    long heapSize = Runtime.getRuntime().totalMemory(); 
    long heapMaxSize = Runtime.getRuntime().maxMemory();
    long heapFreeSize = Runtime.getRuntime().freeMemory();

    System.out.format("Total: [%s], Max: [%s], Free: [%s]\n", heapSize, heapMaxSize, heapFreeSize);
    System.out.format("getMemoryUnit(): Total: [%s], Max: [%s], Free: [%s]\n",
            getMemoryUnit(heapSize), getMemoryUnit(heapMaxSize), getMemoryUnit(heapFreeSize));
    System.out.format("getFullMemoryUnit(): Total: [%s], Max: [%s], Free: [%s]\n",
            getFullMemoryUnit(heapSize), getFullMemoryUnit(heapMaxSize), getFullMemoryUnit(heapFreeSize));
}

ไบต์เพื่อรับรูปแบบข้างต้น

Total: [128974848], Max: [1884815360], Free: [126248240]

เพื่อที่จะแสดงเวลาในรูปแบบที่มนุษย์อ่านได้ใช้ฟังก์ชั่นmillisToShortDHMS(long duration)นี้


0

นี่คือการแปลงจาก @aioobe แปลงเป็น kotlin

/**
 * https://stackoverflow.com/a/3758880/1006741
 */
fun Long.humanReadableByteCountBinary(): String {
    val b = when (this) {
        Long.MIN_VALUE -> Long.MAX_VALUE
        else -> abs(this)
    }
    return when {
        b < 1024L -> "$this B"
        b <= 0xfffccccccccccccL shr 40 -> "%.1f KiB".format(Locale.UK, this / 1024.0)
        b <= 0xfffccccccccccccL shr 30 -> "%.1f MiB".format(Locale.UK, this / 1048576.0)
        b <= 0xfffccccccccccccL shr 20 -> "%.1f GiB".format(Locale.UK, this / 1.073741824E9)
        b <= 0xfffccccccccccccL shr 10 -> "%.1f TiB".format(Locale.UK, this / 1.099511627776E12)
        b <= 0xfffccccccccccccL -> "%.1f PiB".format(Locale.UK, (this shr 10) / 1.099511627776E12)
        else -> "%.1f EiB".format(Locale.UK, (this shr 20) / 1.099511627776E12)
    }
}
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.