程序員都很懶,你懂的!
java程序員在實際的開發中會遇到很多的單位換算問題。今天我給大家帶來的是關於計算機硬盤大小的換算。多數情況下,一般要求 b,kb,mb,gb,tb,pb之間的大小轉換,我們都知道他們之間的換算是乘以1024或者除以1024。但是具體怎么用java代碼來實現呢?請看 下面的代碼:
1 package com.herman.utils; 2 3 /*** 4 * @see 存儲大小(單位)轉換器. 5 * @author Herman.Xiong 6 * @date 2014年5月27日 13:27:40 7 * @version V1.0 8 */ 9 public enum SizeConverter { 10 /** 轉換任意單位的大小, 返回結果會包含兩位小數但不包含單位. */ 11 Arbitrary { 12 @Override 13 public String convert(float size) { 14 while (size > 1024) { 15 size /= 1024; 16 } 17 return String.format(FORMAT_F, size); 18 } 19 }, 20 21 // ----------------------------------------------------------------------- 22 // 有單位 23 /** 轉換單位為B的大小, 返回結果會包含兩位小數以及單位. 如: 1024B->1KB, (1024*1024)B->1MB */ 24 B { 25 @Override 26 public String convert(float B) { 27 return converter(0, B); 28 } 29 }, 30 /** 轉換單位為B的大小, 返回結果會包含兩位小數以及單位. */ 31 KB { 32 @Override 33 public String convert(float KB) { 34 return converter(1, KB); 35 } 36 }, 37 /** 轉換單位為MB的大小, 返回結果會包含兩位小數以及單位. */ 38 MB { 39 @Override 40 public String convert(float MB) { 41 return converter(2, MB); 42 } 43 }, 44 /** 轉換單位為GB的大小, 返回結果會包含兩位小數以及單位. */ 45 GB { 46 @Override 47 public String convert(float GB) { 48 return converter(3, GB); 49 } 50 }, 51 /** 轉換單位為TB的大小, 返回結果會包含兩位小數以及單位. */ 52 TB { 53 @Override 54 public String convert(float TB) { 55 return converter(4, TB); 56 } 57 }, 58 59 // ----------------------------------------------------------------------- 60 // trim沒單位 61 /** 轉換任意單位的大小, 返回結果小數部分為0時將去除兩位小數, 不包含單位. */ 62 ArbitraryTrim { 63 @Override 64 public String convert(float size) { 65 while (size > 1024) { 66 size /= 1024; 67 } 68 69 int sizeInt = (int) size; 70 boolean isfloat = size - sizeInt > 0.0F; 71 if (isfloat) { 72 return String.format(FORMAT_F, size); 73 } 74 return String.format(FORMAT_D, sizeInt); 75 } 76 }, 77 78 // ----------------------------------------------------------------------- 79 // trim有單位 80 /** 轉換單位為B的大小, 返回結果小數部分為0時將去除兩位小數, 會包含單位. */ 81 BTrim { 82 @Override 83 public String convert(float B) { 84 return trimConverter(0, B); 85 } 86 }, 87 /** 轉換單位為KB的大小, 返回結果小數部分為0時將去除兩位小數, 會包含單位. */ 88 KBTrim { 89 @Override 90 public String convert(float KB) { 91 return trimConverter(1, KB); 92 } 93 }, 94 /** 轉換單位為MB的大小, 返回結果小數部分為0時將去除兩位小數, 會包含單位. */ 95 MBTrim { 96 @Override 97 public String convert(float MB) { 98 return trimConverter(2, MB); 99 } 100 }, 101 /** 轉換單位為GB的大小, 返回結果小數部分為0時將去除兩位小數, 會包含單位. */ 102 GBTrim { 103 @Override 104 public String convert(float GB) { 105 return trimConverter(3, GB); 106 } 107 }, 108 /** 轉換單位為TB的大小, 返回結果小數部分為0時將去除兩位小數, 會包含單位. */ 109 TBTrim { 110 @Override 111 public String convert(float TB) { 112 return trimConverter(4, TB); 113 } 114 }; 115 /*** 116 * <p> 將指定的大小轉換到1024范圍內的大小. 注意該方法的最大單位為PB, 最小單位為B, 117 * 任何超出該范圍的單位最終會顯示為**. </p> 118 * 119 * @param size 要轉換的大小, 注意是浮點數, 不要以整形的方式傳入, 容易造成溢出. 120 * (如: 1024*1024*1024*1024*1024會溢出, 使結果為0, 因為它先將結果以int相乘后再轉換為float; 121 * 而1024.0F*1024.0F*1024.0F*1024.0F*1024.0F就不會溢出) 122 * @return 123 */ 124 abstract public String convert(float size); 125 126 // ----------------------------------------------------------------------- 127 // 單位轉換 128 129 private static final String[] UNITS = new String[] { 130 "B", "KB", "MB", "GB", "TB", "PB", "**" 131 }; 132 133 private static final int LAST_IDX = UNITS.length-1; 134 135 private static final String FORMAT_F = "%1$-1.2f"; 136 private static final String FORMAT_F_UNIT = "%1$-1.2f%2$s"; 137 138 private static final String FORMAT_D = "%1$-1d"; 139 private static final String FORMAT_D_UNIT = "%1$-1d%2$s"; 140 141 // ----------------------------------------------------------------------- 142 private static String converter(int unit, float size) { 143 int unitIdx = unit; 144 while (size > 1024) { 145 unitIdx++; 146 size /= 1024; 147 } 148 int idx = unitIdx < LAST_IDX ? unitIdx : LAST_IDX; 149 return String.format(FORMAT_F_UNIT, size, UNITS[idx]); 150 } 151 152 private static String trimConverter(int unit, float size) { 153 int unitIdx = unit; 154 while (size > 1024) { 155 unitIdx++; 156 size /= 1024; 157 } 158 159 int sizeInt = (int) size; 160 boolean isfloat = size - sizeInt > 0.0F; 161 int idx = unitIdx < LAST_IDX ? unitIdx : LAST_IDX; 162 if (isfloat) { 163 return String.format(FORMAT_F_UNIT, size, UNITS[idx]); 164 } 165 return String.format(FORMAT_D_UNIT, sizeInt, UNITS[idx]); 166 } 167 168 // ----------------------------------------------------------------------- 169 public static String convertBytes(float B, boolean trim) { 170 return trim ? trimConvert(0, B, true) : convert(0, B, true); 171 } 172 173 public static String convertKB(float KB, boolean trim) { 174 return trim ? trimConvert(1, KB, true) : convert(1, KB, true); 175 } 176 177 public static String convertMB(float MB, boolean trim) { 178 return trim ? trimConvert(2, MB, true) : convert(2, MB, true); 179 } 180 181 /*** 182 * <p> 存儲大小單位間的轉換. 注意該方法的最大單位為PB, 最小單位為B, 183 * 任何超出該范圍的單位最終會顯示為**. </p> 184 * 185 * @param unit 從哪個單位開始 186 * @param size 存儲大小, 注意是float, 不要以整形的形式傳入, 否則會溢出(如:1024*1024這種, 187 * 它是先將1024*1024作為int相乘再轉換為float的, 如果值過大的話就會溢出了, 188 * 所以這么寫1024.0F*1024.0F) 189 * @param withUnit 返回的結果字符串是否帶有對應的單位 190 * @return 191 */ 192 private static String convert(int unit, float size, boolean withUnit) { 193 int unitIdx = unit; 194 while (size > 1024) { 195 unitIdx++; 196 size /= 1024; 197 } 198 if (withUnit) { 199 int idx = unitIdx < LAST_IDX ? unitIdx : LAST_IDX; 200 return String.format(FORMAT_F_UNIT, size, UNITS[idx]); 201 } 202 return String.format(FORMAT_F, size); 203 } 204 205 /*** 206 * <p> 存儲大小單位間的轉換, 如果轉換后小數部分為0, 則去除小數部分. 207 * 注意該方法的最大單位為PB, 最小單位為B, 任何超出該范圍的單位最終會顯示為**. </p> 208 * 209 * @param unit 從哪個單位開始 210 * @param size 存儲大小, 注意是float, 不要以整形的形式傳入, 否則會溢出(如:1024*1024這種, 211 * 它是先將1024*1024作為int相乘再轉換為float的, 如果值過大的話就會溢出了, 212 * 所以這么寫1024.0F*1024.0F) 213 * @param withUnit 返回的結果字符串是否帶有對應的單位 214 * @return 215 */ 216 private static String trimConvert(int unit, float size, boolean withUnit) { 217 int unitIdx = unit; 218 while (size > 1024) { 219 unitIdx++; 220 size /= 1024; 221 } 222 223 int sizeInt = (int) size; 224 boolean isfloat = size - sizeInt > 0.0F; 225 if (withUnit) { 226 int idx = unitIdx < LAST_IDX ? unitIdx : LAST_IDX; 227 if (isfloat) { 228 return String.format(FORMAT_F_UNIT, size, UNITS[idx]); 229 } 230 return String.format(FORMAT_D_UNIT, sizeInt, UNITS[idx]); 231 } 232 233 if (isfloat) { 234 return String.format(FORMAT_F, size); 235 } 236 return String.format(FORMAT_D, sizeInt); 237 } 238 }
工具類代碼寫好了,我們來看一個測試類吧,上代碼:
1 package com.herman.test; 2 3 import com.herman.utils.SizeConverter; 4 /** 5 * @see 硬盤大小換算測試類 6 * @author Herman.Xiong 7 * @date 2014年5月27日 13:43:33 8 */ 9 public class SizeConverterTest { 10 public static void main(String[] args) { 11 System.out.println(SizeConverter.MBTrim.convert(419562f)); 12 } 13 }