由于所有Java浮点数,即floats和doubles,都以位形式内部表示,因此我希望找到一种高效的算法,将表示该float或double的比特串转换为相应的浮点数 - 我找不到内置的库函数,所以只能自己编写。
长度为32的二进制字符串表示float,而长度为64的二进制字符串将转换为double。所有floats都可以无损精度地转换为doubles。空格会被忽略。
示例
"0 10000000 10010010000111111011011"
成为3.141592
作为float。"1 11111111 00000000000000000000000"
成为-infinity
。"0 11111111 10010010000111111011011"
成为一个floatNaN
。"1 10000000000 0101101111110000101010001011000101000101011101101001"
成为最接近-e的double值,即2.71828182845904509079559829843
到目前为止,我有这么多的代码:
public static double ieee(String binString) throws Exception {
binString = binString.replace(" ", "");
if (binString.length() == 32) {
String exponentB = binString.substring(1, 9);
String mantissaB = binString.substring(9, 32);
int sgn = binString.charAt(0) == '0' ? 1 : -1;
int exponent = Integer.parseInt(exponentB, 2) - 127; // Biased by 127
double mantissa = 1 + Integer.parseInt(mantissaB, 2) / Math.pow(2, 23);
if (exponent == 128 && mantissa == 1)
return sgn == 1 ? Double.POSITIVE_INFINITY : Double.NEGATIVE_INFINITY;
if (exponent == 128 && mantissa != 0)
return Double.NaN;
if (exponent == -127)
return sgn*Math.pow(2,-126)*(mantissa - 1);
return sgn*Math.pow(2, exponent)*mantissa;
}
else if (binString.length() == 64) {
String exponentB = binString.substring(1, 12);
String mantissaB = binString.substring(12, 64);
int sgn = binString.charAt(0) == '0' ? 1 : -1;
int exponent = Integer.parseInt(exponentB, 2) - 1023; // Biased by 1023
double mantissa = 1 + Long.parseLong(mantissaB, 2) / Math.pow(2, 52);
if (exponent == 1024 && mantissa == 1)
return sgn == 1 ? Double.POSITIVE_INFINITY : Double.NEGATIVE_INFINITY;
if (exponent == 1024 && mantissa != 0)
return Double.NaN;
if (exponent == -1023)
return sgn*Math.pow(2,-1022)*(mantissa - 1);
return sgn*Math.pow(2, exponent)*mantissa;
}
else {
throw new Exception("Does not represent internal bits of a floating-point number");
}
}
尽管我的代码目前能够正常工作,但在速度和代码量方面,将IEEE-754二进制表示字符串转换为其对应的float
或double
的最简洁或最快速的方法是什么?最有效的方法以及其效率和专业知识的良好解释是首选。