为什么会发生以下情况:
char p = 0;
p--;
System.out.println(p);
Run Code Online (Sandbox Code Playgroud)
结果65535
为什么不给出编译错误或运行时异常?我期待它作为字符不能是负面的.相反,它从倒置开始倒计时.提前致谢.
Dan*_*her 16
为什么不给出编译错误或运行时异常?
因为语言规范要求对原始类型的算术是模数2^width,所以-1变为2^16-1a char.
在关于整数运算的部分中,说明了这一点
内置整数运算符不以任何方式指示溢出或下溢.
所以禁止抛出异常.
对于所使用的postfix-decrement运算符,具体来说,其行为在15.14.3中指定
否则,从变量的值中减去值1,并将差值存储回变量中.在减法之前,对值1和变量的值执行二进制数字提升(第5.6.2节).如有必要,通过缩小基元转换(第5.1.3节)和/或在存储之前对变量类型进行装箱转换(第5.1.7节)来缩小差异.后缀减量表达式的值是存储新值之前变量的值.
二进制数字提升将值和值1转换为int(因为这里的类型是char),因此您将中间结果-1作为a int,然后执行缩小基元转换:
将有符号整数缩小到整数类型T只会丢弃除n个最低位之外的所有位,其中n是用于表示类型T的位数.除了可能丢失有关数值大小的信息之外,这可能导致结果值的符号与输入值的符号不同.
得到一个char值0xFFFF(因为Java为其有符号整数类型指定了两个补码表示,在一元减去规范中明确说明):
对于整数值,否定与从零减去相同.Java编程语言对整数使用二进制补码表示,并且二进制补码值的范围不对称,因此否定最大负数int或long会导致相同的最大负数.在这种情况下会发生溢出,但不会抛出异常.对于所有整数值x,-x等于(~x)+1.
对于超出范围结果的一般环绕行为,作为乘法运算符规范中的示例:
如果整数乘法溢出,则结果是数学乘积的低阶位,如某些足够大的二进制补码格式所示.结果,如果发生溢出,则结果的符号可能与两个操作数值的数学乘积的符号不同.
类似的短语出现在整数加法的规范中,并且需要减法来实现a - b == a + (-b),因此溢出行为如下.