负char值JAVA

为什么会发生以下情况:

char p = 0; p--; System.out.println(p); 

结果65535

为什么不给出编译错误或运行时exception? 我期待它作为字符不能是负面的。 相反,它从倒置开始计数。 提前致谢。

为什么不给出编译错误或运行时exception?

因为语言规范要求对基本类型的算术是模2^width ,所以-1作为char变为2^16-1

在关于整数运算的部分中 ,说明了这一点

内置整数运算符不以任何方式指示上溢或下溢。

所以禁止抛出exception。

对于所使用的postfix-decrement运算符,具体来说,其行为在15.14.3中指定

否则,从变量的值中减去值1,并将差值存储回变量中。 在减法之前,对值1和变量的值执行二进制数字提升(第5.6.2节)。 如有必要,通过缩小基元转换(第5.1.3节)和/或在存储之前对变量类型进行装箱转换(第5.1.7节)来缩小差异。 postfix减量表达式的值是存储新值之前的变量值。

二进制数字提升将值和值1转换为int (因为此处的类型为char ),因此您将中间结果-1作为int ,然后执行缩小基元转换:

将有符号整数缩小到整数类型T只会丢弃除n个最低位之外的所有位,其中n是用于表示类型T的位数。除了可能丢失有关数值大小的信息之外,这可能导致结果值的符号与输入值的符号不同。

导致char值为0xFFFF (因为Java为其有符号整数类型指定了两个补码表示,在一元减去规范中明确说明):

对于整数值,否定与从零减去相同。 Java编程语言对整数使用二进制补码表示 ,并且二进制补码值的范围不对称,因此否定最大负数int或long会导致相同的最大负数。 在这种情况下会发生溢出,但不会抛出exception。 对于所有整数值x,-x等于(~x)+1。

对于超出范围结果的一般环绕行为,作为乘法运算符规范中的示例:

如果整数乘法溢出,则结果是数学乘积的低阶位,如某些足够大的二进制补码格式所示。 结果,如果发生溢出,则结果的符号可能与两个操作数值的数学乘积的符号不同。

类似的短语出现在整数加法的规范中,并且需要减法来满足a - b == a + (-b) ,因此溢出行为如下。

因为这就是Java语言的定义方式。 运行时不检查每个操作的边界(可能因为它非常昂贵)。 它只是溢出或下溢。