java - Static 標識的字段或者是代碼塊,真的是在類加載的時候初始化的嗎?
問題描述
class AAA { static {System.out.println('class AAA static block println'); // 并沒有打印此句 }}public class Main { public static void main(String[] args) {System.out.println('hello world!'); }}
一直以來都以為 static 標識的代碼塊或者是字段,都是在類加載的時候就被執行或者賦值了,但是這么一看....感覺自己的世界觀都要被刷新了。
所以此處是類沒有被加載嗎?還是說我們一直以來認為的,靜態代碼塊、字段都在類加載的時候被初始化的,這個觀點是錯誤的?
在《深入理解Java虛擬機:JVM高級特性與最佳實踐 第2版》中找到一些線索,如下圖:
所以,照這么說,是在第一次主動訪問該類的時候執行?小弟好生迷惑啊....大家快說說你們的觀點
問題解答
回答1:類初始化和對象初始化。
static包含的代碼塊和變量只有在類初始化的時候才執行,而初始化的五種條件你也知道啦。
補充說明清楚吧。首先,你即使放在同一個.java文件中,編譯后,這還是兩個不同的class文件,不信你看看bin對應的包下面生成的.class文件。第二,類初始化的時候,就會初始化類的靜態變量和運行靜態代碼塊。所以,虛擬機規定了五種初始化的條件,比如使用了new,getstatic,putstatic指令,main函數所在的類,反射,父類等情況。而,除開這五種情況,是不能觸發類的初始化的。正如你代碼中所示Main.class中,并沒有任何關于AAA.class的調用或者父子關系或者反射。所以,AAA.class自然不會初始化。
可以看看的另一篇博客java類的加載過程
明白了嗎?
回答2:-XX:+TraceClassLoading加上這個會發現沒加載AAA
回答3:這里有兩個概念需要擼一下:
類加載機制
Java、編譯器、字節碼、JVM的規范和實現。
類的加載是通過類加載器(Classloader)完成的,加載的具體策略依賴JVM的具體實現,總的來說可以分兩種:
饑餓式加載,只要被其他類引用到了就加載。
懶惰式加載,當類被訪問的時候才加載。
Java、編譯器、字節碼、JVM有各自的規范,彼此通過規范協同工作:
編譯按把Java代碼編譯成規范的字節碼文件,每一個類(外部類、內部類、匿名類)都會被編譯成一個單獨的字節碼文件(class文件),JVM加載類的時候就是從這些class文件中一個個的加載。
現在回到你的代碼:
在Java層,你把AAA、Main兩個類放在一個文件中,編譯器編譯后生成兩個class文件:AAA.class、Main.class。兩個類在代碼組織形式上是一起的,但是編譯后卻是獨立的,并且Main并沒有引用AAA,所以無論是哪種類加載方式都不會觸發對類AAA的加載,也就不會執行AAA中的靜態代碼塊。
回答4:真心感謝樓上熱心網友們的解答!
驗證AAA 類確實沒有被加載,只有 Main 類被加載(題干截圖:初始化條件第四條,主類被 jvm 自動加載)
java -XX:+TraceClassLoading Main結論
類中 靜態字段|代碼塊 真的是在類加載的時候被初始化或者是執行的!
延伸怎么知道類有沒有被 jvm 所加載?這也是我一直糾結的問題,一開始以為只要執行了 javac 命令,類就被 jvm 加載了,其實不然,該命令只是將 .java 文件轉化成 jvm 能讀懂的 .class 文件而已。
那么到底怎么知道類有沒有被 jvm 所加載?據 《深入理解Java虛擬機:JVM高級特性與最佳實踐 第2版》 和廣大網友的熱心解答可知,并沒有明確的時機規定了啥時候會被加載!
但是!jvm 明確規定了類被初始化的時機-就是題干上截圖部分那四種!而類的加載是優先于類初始化的,所以這里,我們暫且可以認為這幾種情況就是觸發類加載的條件。
小弟愚昧,總結不妥之處,還麻煩大家指正!感謝
回答5:把你的Main.java和AAA.java放在同一個文件夾里,
在main函數里寫
Class.forName('AAA');
執行
回答6:執行main方法時,只會加載Main類,Main類中并沒有使用到AAA類,并不會去加載AAA類,并不是說把AAA和Main兩個類寫到同一個文件就會同時加載
回答7:AAA這個類既沒有在其他地方new,也沒有對應的去獲取或者設置靜態的字段,也沒有去invoke靜態方法。所以不會自動初始化的。
回答8:放在兩個類里面了,聲明為public的類中的mian開始執行,那個類沒被用到自然不會被加載更別提初始化
