根据 IEEE 754 ,指数位全为 1 且尾数位不全为 0 的浮点数为 NaN 。所以 NaN 应该有多种二进制形式(尾数位有多种情况),然而,我使用下面的程序进行测试,打印出来的确是 1111111110000000000000000000000 。为什么会这样?
public class Test {
public static String toBinary(String bits){
float f = Float.intBitsToFloat(Integer.parseInt(bits, 2));
int i = Float.floatToIntBits(f);
String s = Integer.toBinaryString(i);
return s;
}
public static void main(String[] args) {
System.out.println(Test.toBinary("01111111100000000000000000000001"));
}
}
一个有点怪异的问题...
1
imKiva 2021-11-14 18:57:12 +08:00
`Float.intBitsToFloat` 的[文档]( https://docs.oracle.com/javase/8/docs/api/java/lang/Float.html#intBitsToFloat-int-)中有一句:`If the argument is any value in the range 0x7f800001 through 0x7fffffff or in 0xff800001 through 0xffffffff, the result is a NaN.`
`Float.floatToIntBits` 的[文档]( https://docs.oracle.com/javase/8/docs/api/java/lang/Float.html#floatToIntBits-float-)中有一句:`If the argument is NaN, the result is 0x7fc00000.` |
2
icyalala 2021-11-14 19:04:39 +08:00
看一下 floatToIntBits 源码,里面会做判断:如果是 NaN ,就使用固定值( 0x7fc00000 )。
NaN 那段无效位叫 payload , 可以用于存放一些诊断信息或者平台特有数据(例如 NaN-boxing ),但是现在基本上没有实际使用的场景。 |
3
Higurashi OP |
4
agagega 2021-11-14 20:02:32 +08:00 via iPhone
补充一下,在一些平台上会有 signaling nan 和 quiet nan 的区分,区别是指令处理到 singaling nan 会触发 nan singal 异常。这两种 nan 就是按二楼说的 payload 来区分的
|
5
icyalala 2021-11-14 20:13:09 +08:00
@Higurashi
byte[] bytes = new byte[4]; ByteBuffer.wrap(bytes).putFloat(f); int i = ByteBuffer.wrap(bytes).getInt(); |
7
Higurashi OP |