编译以下类的时候,如果切换WARNING_LEVEL的赋值方式为"= StatusDef.WARNING",javac会报“非法的前向引用”。 Eclipse自带的编译器会编译通过,这是Eclipse编译器的一个缺陷。
产生这个编译错误的原因是,cinit方法是按照字段在源码中的声明顺序依次执行各个初始化代码段。 对于ForwardReference类,cinit方法依次执行以下功能:
- 通过NORMAL_LEVEL的数值,创建C0000_1字段。 在ForwardReference.class文件中,NORMAL_LEVEL字段的初始值是写在字段的attribute属性中的, 因此在进入cinit方法以前,NORMAL_LEVEL字段已经被设置好了初始值,即0x32.
- 通过WARNING_LEVEL的数值,创建C0000_2字段。但是,此时WARNING_LEVEL字段的数值还是0。
- 把WARNING_LEVEL字段的数值设置为StatusDef.WARNING。 可以看到,第二步创建的C0000_2并没有包含开发者期望的结果。
public enum ForwardReference { C0000_1 (ForwardReference.NORMAL_LEVEL), C0000_2 (ForwardReference.WARNING_LEVEL); public static final short NORMAL_LEVEL = 0x32; // 切换以下注释行,以在javac下产生”非法前向引用的错误“ // public static final short WARNING_LEVEL = StatusDef.WARNING; public static final short WARNING_LEVEL = 0x08; private final short statusLevel; public short getStatusLevel() { return statusLevel; } private ForwardReference(short statusLevel){ this.statusLevel = statusLevel; } public static void main(String[] args){ System.out.println(ForwardReference.C0000_1.getStatusLevel()); System.out.println(ForwardReference.C0000_2.getStatusLevel()); } } class StatusDef { public final static Short WARNING = 0x08; }
时间: 2025-01-05 06:31:26