關於編寫Java程序讓Jvm崩潰


今天在書上看到一個作者提出一個問題“怎樣通過編寫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 這樣的錯誤。如果想使他們棧空間不足,最簡單的,就是在方法里,如構造方法里不斷申請新的內存空間就夠了,如我第二個錯誤例子的示范。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM