常量的本质含义与反编译及助记符详解
public class FinalClass {
public static void main(String[] args) {
System.out.println(MyParent2.str);
}
}
class MyParent2{
public static String str = "hello world";
static {
System.out.println("MyParent static block");
}
}
当public static String str = "hello world"时,(没有final修饰)
运行结果为:
MyParent static block
hello world
当public static final String str = "hello world"时,(有final修饰)
运行结果为:
hello world
- final修饰的这个常量str在编译阶段会被存入到 [ 调用这个常量的方法 (所在的类) ]的常量池当中;
- 本质上,调用类[这里是:FinalClass]并没有直接引用到定义常量的类[这个是:MyParent2],因此并不会触发定义常量的类的初始化。因为MyParent2.str不会触发MyParent2的初始化,所以MyParent2中的静态代码块就不会被执行。
- 注意:这里指的是将常量存放到了FinalClass的常量池中,之后FinalClass与MyParent2就没有任何关系了。甚至,我们可以将MyParent2的class文件删除。
编译与反编译结果:
E:\Program Files (x86)\IdeaProject\正则表达式啊\out\production\IdeaProject\Jvm>javap FinalClass.class
Compiled from "FinalClass.java"
public class Jvm.FinalClass {
public Jvm.FinalClass();
public static void main(java.lang.String[]);
}
E:\Program Files (x86)\IdeaProject\正则表达式啊\out\production\IdeaProject\Jvm>javap -c FinalClass.class
Compiled from "FinalClass.java"
public class Jvm.FinalClass {
public Jvm.FinalClass();
Code:
0: aload_0
1: invokespecial #1
4: return
public static void main(java.lang.String[]);
Code:
0: getstatic #2
3: ldc #4
5: invokevirtual #5
8: return
}
public static String str1 = "hello world"; ----> ldc
public static short str2 = 7; ----> bipush
public static short str3 = 127; ----> bipush
public static int str4 = 128; ----> sipush
public static int str5 = 1; ----> iconst_1
public static int str6 = 2; ----> iconst_2
ldc:表示将int,float,String类型的常量值从常量池中推送至栈顶
bipush:表示将单字节 -128 ~ 127的常量值推送至栈顶
sipush:表示将短整型的常量值 -32768 ~ 32767 推送至栈顶
iconst_1:表示将 int 类型的1推送至栈顶 (iconst_m1 ~ iconst_5)