標籤:java
編譯以下類的時候,如果切換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;}
Javac編譯時間報“非法的前向引用”,以及Eclipse內建的Java編譯器的缺陷