【问题标题】:Numeric assignment should throw error数值赋值应该抛出错误
【发布时间】:2011-06-10 17:33:48
【问题描述】:

我正在用 Java 中的数字转换和强制转换进行一些测试,我发现了这种奇怪的行为(对我来说)

class Test {
public static void main(String[] args) {
    //int y = 100000000000000; //does not compile
    int x = 100000 * 1000000000; //compile (why?)
    System.out.println(x); //x is 276447232
   }
}

x 和 y 基本上应该是同一个数:为什么会编译?

【问题讨论】:

    标签: java variable-assignment numeric


    【解决方案1】:

    在 Java 中未检测到整数溢出,这就是乘法工作正常的原因。但是,您所说的文字太大,因此是编译器错误。

    我猜大多数 Java 编译器会在编译时预先计算 JLS 不需要的值。因此它不可能是错误,因为不同的编译器可能会编译一段代码或产生错误——这不是那么好。

    Sun Java 编译器实际上执行了反汇编所示的计算:

    Compiled from "x.java"
    class x extends java.lang.Object{
    x();
      Code:
       0:   aload_0
       1:   invokespecial   #1; //Method java/lang/Object."<init>":()V
       4:   return
    
    public static void main(java.lang.String[]);
      Code:
       0:   ldc     #2; //int 276447232
       2:   istore_1
       3:   getstatic       #3; //Field java/lang/System.out:Ljava/io/PrintStream;
       6:   iload_1
       7:   invokevirtual   #4; //Method java/io/PrintStream.println:(I)V
       10:  return
    

    这里要注意的重要一点是,对于所有意图和目的,结果必须与在运行时计算的结果相同。因此这里不会出现编译错误。

    【讨论】:

    • 请注意,在某些情况下它必需的——例如在“case”标签中使用值。另请注意,作为常量表达式可能会更改较大表达式的整体类型。
    • 乔恩:随时添加(在此或其他答案中)。关于这个话题,我不能说更多。我偶尔会略读 JLS,但我的语言不是很深,抱歉。
    【解决方案2】:

    Java 不是通过异常处理溢出整数,而是通过翻转为负值并继续向上。

    作为一个例子,这里的代码是 sn-p:

    int x = Integer.MAX_VALUE + 1;
    System.out.println(x);
    

    Java 尽其所能地处理它。那个 sn-p 输出:

    -2147483648
    

    这是 Integer.MIN_VALUE 的值。当你处理非常大的数字时,你只需要小心。这就是为什么 Java 有一个 BigInteger 类来处理超出整数限制的任意大数字。

    【讨论】:

      猜你喜欢
      • 2020-01-08
      • 2021-11-13
      • 1970-01-01
      • 1970-01-01
      • 2019-04-14
      • 1970-01-01
      • 2012-08-18
      • 2011-05-06
      • 2018-02-02
      相关资源
      最近更新 更多