Integer.valueOf转换负数时抛出NumberFormatException

来源:互联网 发布:java script else 编辑:程序博客网 时间:2024/06/05 20:44

1. 问题描述

今天在牛客上,有牛友抛出了这样一个问题:


Integer m =Integer.valueOf("11111111111111111111111100000000",2);

上面这段代码,二进制数的长度是32位,没超出Integer的范围,为什么会抛出NumberFormatException的异常呢?

2. 探究

当时我也很好奇,所以就打开IDE,探究了一下源码,以下是过程:
这里写图片描述
下面,先看一下parseInt的注释:
这里写图片描述
注释中说,是通过查看第0个字符是否为’-‘来判定负数,而不是看最高位是否为1。如此一来好像解释的通了。再看一下真正的转换过程(这里我调了一下源码的排版,不然惨不忍睹…):
这里写图片描述
确实如注释中所说,此时,一个32位且最高位为1的二进制数,会被错误的判定为正数。而在下面有这样一行代码:
这里写图片描述
可以看到,一个32位且最高位为1的二进制数,会被累加32次,所以其累加和是超过Integer正数的范围的。实际上不用等到第二个红框位置,在if(result<multmin)时,就已经抛出异常了。multmin我觉得为了防止result*radix后溢出而设置的。代码的运行结果也印证了这个推测:
这里写图片描述
这里写图片描述
另外,result每次是加上负的digit,这是因为Integer的表示范围是-2的32次方到2的32次方-1.正数的值要小,那么在计算一个负数时,如果遵循先正向累加再求负的过程,是有可能溢出的。最终的结果输出可以很明显的看到是result是负数:
这里写图片描述

0 0