美文网首页java 语法糖
Java 语法糖(二): 自动装箱

Java 语法糖(二): 自动装箱

作者: jyjz2008 | 来源:发表于2020-07-02 22:24 被阅读0次

    正文

    一个简单的装箱操作

    javac 会自动完成装箱的操作,我用一个名为 Main.java 的程序来演示一下。
    Main.java 的内容如下

    public class Main {
        private Integer f(int in) {
            return in;
        }
    }
    

    执行如下命令对 Main.java 进行编译

    javac Main.java
    

    执行如下命令可以查看字节码文件中的内容

    javap -cp . -p -v 'Main'
    

    完整的内容较长,和 f(...) 直接相关的部分如下

      private java.lang.Integer f(int);
        descriptor: (I)Ljava/lang/Integer;
        flags: ACC_PRIVATE
        Code:
          stack=1, locals=2, args_size=2
             0: iload_1
             1: invokestatic  #2                  // Method java/lang/Integer.valueOf:(I)Ljava/lang/Integer;
             4: areturn
          LineNumberTable:
            line 3: 0
    

    f(...) 对应的字节码指令有三条, 作用如下

    1. iload_1, 其作用是将1slot中的int值加载到操作数栈中(f(...)方法只有一个参数in1slot中保存的就是 in 这个参数的值)
    2. invokestatic #2, 调用常量池中与 #2 对应的静态方法, 也就是 java/lang/Integer.valueOf:(I)Ljava/lang/Integer; 这个方法
    3. areturn, 将操作数栈栈顶的引用作为 f(...) 的返回值

    显式地进行装箱

    大致可以猜到,编译器在遇到需要对 int 类型进行装箱的场景时,会自动调用 Integer 类中的 valueOf(...) 方法

    我们可以写一个 Temp.java 来验证一下

    public class Temp {
        private Integer f(int in) {
            return Integer.valueOf(in);
        }
    }
    

    执行如下命令就能看到 Temp.java 中的 f(...) 对应的字节码指令

    javac Temp.java
    javap -cp . -p -v 'Temp'
    

    Temp.java 中的 f(...) 的字节码指令展示如下

      private java.lang.Integer f(int);
        descriptor: (I)Ljava/lang/Integer;
        flags: ACC_PRIVATE
        Code:
          stack=1, locals=2, args_size=2
             0: iload_1
             1: invokestatic  #2                  // Method java/lang/Integer.valueOf:(I)Ljava/lang/Integer;
             4: areturn
          LineNumberTable:
            line 3: 0
    

    可见 Main.java Temp.java 中的 f(...) 在编译后生成的字节码指令确实是相同的

    也可以用 cfr 这个工具验证一下。
    cfr-0.144.jar 为例,执行如下命令,就能看到显式的装箱操作

    java -jar cfr-0.144.jar Main --sugarboxing false 
    

    运行结果如下

    /*
     * Decompiled with CFR 0.144.
     */
    public class Main {
        private Integer f(int n) {
            return Integer.valueOf((int)n);
        }
    }
    

    可见,cfr 运行的结果和我们之前的猜测是一致的(除了那个多余的类型转换之外)。

    参考文章

    1. https://www.benf.org/other/cfr/boxing-isnt-magic.html

    相关文章

      网友评论

        本文标题:Java 语法糖(二): 自动装箱

        本文链接:https://www.haomeiwen.com/subject/fimdqktx.html