今天在書上看到一個作者提出一個問題“怎樣通過編寫Java代碼讓Jvm崩潰”,我看了之后也不懂。帶着問題查了一下,百度知道里面有這樣一個答案:
1 package jvm; 2 3 public class Crash { 4 public static void main(String[] args) { 5 6 //Object[] o = {“abc”};初始值賦值,不會有影響。 7 Object[] o = null; 8 9 while (true) { 10 o = new Object[] { o }; 11 //輸出的話,jvm就不會崩潰。 12 //System.out.println(o); 13 } 14 } 15 }
程序運行十幾秒之后,控制台會出現這樣的錯誤:
Exception in thread "main" java.lang.OutOfMemoryError: Java heap space
at jvm.Crash.main(Crash.java:10)
很明顯,超出內存空間錯誤。
我將原程序隨意改了一下,如賦初始值等,對程序無影響。
可是我將死循環中的o輸出在控制台的時候,jvm居然一直都不崩,為什么輸出的話,就不會超出內存空間呢?
我看來,原程序能夠使Jvm崩潰,是因為死循環中,通過舊對象,不斷創建出新的對象,即創造的對象是互相引用的,所以GC是不會回收它們的,造成堆棧溢出。
仿照這個例子,我寫了一個簡單的類,模仿例子程序中的數組,如下:
1 package jvm; 2 3 public class JvmBean { 4 5 JvmBean bean = new JvmBean(this); 6 7 public JvmBean(JvmBean bean){ 8 this.bean = bean; 9 } 10 }
然后簡單測試,如下:
1 package jvm; 2 3 public class MyCrash { 4 5 public static void main(String[] args) { 6 JvmBean j = null; 7 while(true){ 8 j = new JvmBean(j); 9 //無論輸出不輸出,jvm都會崩潰 10 //System.out.println(j); 11 } 12 } 13 }
結果便是控制台輸出如下的錯誤:
Exception in thread "main" java.lang.StackOverflowError at jvm.JvmBean.<init>(JvmBean.java:5) at jvm.JvmBean.<init>(JvmBean.java:5) at jvm.JvmBean.<init>(JvmBean.java:5) at jvm.JvmBean.<init>(JvmBean.java:5) at jvm.JvmBean.<init>(JvmBean.java:5)
一長串的"at jvm.JvmBean.<init>(JvmBean.java:5)",后面的被我省略了。
結果看來,同樣也造成了jvm崩潰,可是錯誤類型跟例子程序的不同,說堆棧溢出錯誤,並且無論是否輸出,錯誤都一樣發生,為什么呢?
由於評論的兩位老兄的熱心指點,兩個問題都水落石出了!
這里過一下整個流程。
第一個異常 結合天添老兄說的,Exception in thread "main" java.lang.OutOfMemoryError: Java heap space at jvm.Crash.main(Crash.java:10)是因為程序無法申請到足夠的內存的時候拋出的異常,Object數組o不斷指向新的Object數組,數組元素是原來的Object數組,這使得Object維數越來越高。不斷申請內存空間,最終導致超出jvm中堆的最大值。堆內存溢出。為什么輸出打印,時間會延長呢?yahokuma老兄一言驚醒夢中人!輸出打印的話,虛擬機並不是不會崩潰,而是崩潰的時間大大延長了。而崩潰時間延長其實是假象,是因為輸出屬於IO事件,每次輸出CPU都被中斷,IO很耗時,所以,感覺上才會時間延長。
第二個異常,yahokuma 老兄在下面評論中已經說的很清楚了,我這里搬過來——“類內部的靜態屬性 > 靜態塊 > 對象屬性 > 構造方法。注意這一點,那就是說 bean屬性會先於JvmBean的構造函數被初始化。在你main函數中,new一個 JvmBean的構造函數之前,類內部的JvmBean對象要優先被初始化,這個類內部的屬性bean的內部同樣也包含了一個JvmBean對象需要被初始化,成循環調用,造 成了棧溢出。”所以異常才會是這個——Exception in thread "main" java.lang.StackOverflowError
我把原JvmBean改一下
1 package jvm; 2 3 public class JvmBean { 4 5 JvmBean bean = null; 6 7 public JvmBean(JvmBean bean){ 8 this.bean = bean; 9 } 10 }
這樣最終得到的結果跟第一個例子一樣了。
如何使Jvm崩潰呢?如果想使它堆內存空間不足,造成典型的內存泄漏,可以創建對象,使它們不斷向深層次引用。產生Exception in thread "main" java.lang.OutOfMemoryError: Java heap space 這樣的錯誤。如果想使他們棧空間不足,最簡單的,就是在方法里,如構造方法里不斷申請新的內存空間就夠了,如我第二個錯誤例子的示范。