案例介紹
這里,優化又溢我將在平時工作過程中總結的納尼內存內存溢出的情況,以代碼案例的候總形式直觀的分享給大家,希望能夠為小伙伴們帶來實質性的結波幫助。
接下來,優化又溢我們就以代碼案例的納尼內存形式來分析各種內存溢出的情況。
定義主類結構
首先,候總我們創建一個類叫做BlowUpJVM,結波所有的優化又溢案例實驗都是基于這個類進行。
public?class?BlowUpJVM?{ ??
}?
棧深度溢出
public?static?void??testStackOverFlow(){ ?
??????BlowUpJVM.testStackOverFlow();?
}?
棧不斷遞歸,納尼內存而且沒有處理,候總所以虛擬機棧就不斷深入不斷深入,棧深度就這樣溢出了。
永久代內存溢出
public?static?void?testPergemOutOfMemory1(){ ?
???//方法一失敗?
????List?list?=?new?ArrayList();?
?
???while(true){ ?
??????list.add(UUID.randomUUID().toString().intern());?
???}?
}?
打算把String常量池堆滿,沒想到失敗了,JDK1.7后常量池放到了堆里,也能進行垃圾回收了。
然后換種方式,使用cglib,用Class把老年代取堆滿
public?static?void?testPergemOutOfMemory2(){ ?
???try?{ ?
??????while?(true)?{ ?
?????????Enhancer?enhancer?=?new?Enhancer();?
?????????enhancer.setSuperclass(OOM.class);?
?????????enhancer.setUseCache(false);?
?????????enhancer.setCallback(new?MethodInterceptor()?{ ?
????????????@Override?
????????????public?Object?intercept(Object?obj,?Method?method,?Object[]?args,?MethodProxy?proxy)?throws?Throwable?{ ?
???????????????return?proxy.invokeSuper(obj,?args);?
????????????}?
?????????});?
?????????enhancer.create();?
??????}?
???}?
???catch?(Exception?e){ ?
??????e.printStackTrace();?
???}?
}?
虛擬機成功內存溢出了,那JDK動態代理產生的類能不能溢出呢?
public?static?void?testPergemOutOfMemory3(){ ?
???while(true){ ?
???final?OOM?oom?=?new?OOM();?
???Proxy.newProxyInstance(oom.getClass().getClassLoader(),?oom.getClass().getInterfaces(),?new?InvocationHandler()?{ ?
?????????public?Object?invoke(Object?proxy,?Method?method,?Object[]?args)?throws?Throwable?{ ?
????????????Object?result?=?method.invoke(oom,?args);?
????????????return?result;?
?????????}?
??????});?
???}?
}?
事實表明,JDK動態代理差生的類不會造成內存溢出,原因是:JDK動態代理產生的類信息,不會放到永久代中,而是放在堆中。
本地方法棧溢出
public?static?void?testNativeMethodOutOfMemory(){ ?
???int?j?=?0;?
???while(true){ ?
??????Printer.println(j++);?
??????ExecutorService?executors?=?Executors.newFixedThreadPool(50);?
??????int?i=0;?
??????while(i++<10){ ?
?????????executors.submit(new?Runnable()?{ ?
????????????public?void?run()?{ ?
????????????}?
?????????});?
??????}?
???}?
}?
這個的原理就是不斷創建線程池,而每個線程池都創建10個線程,這些線程池都是在本地方法區的,久而久之,本地方法區就溢出了。
JVM棧內存溢出
public?static?void?testStackOutOfMemory(){ ?
????while?(true)?{ ???
????????????Thread?thread?=?new?Thread(new?Runnable()?{ ???
???????????????????public?void?run()?{ ?
??????????????????????????while(true){ ?
??????????????????????}?
???????????????????}???
????????????});???
????????????thread.start();???
?????}???
}?
線程的創建會直接在JVM棧中創建,但是本例子中,沒看到內存溢出,主機先掛了,不是JVM掛了,真的是主機掛了,無論在mac還是在windows,都掛了。
溫馨提示,這個真的會死機的。
堆溢出
public?static?void?testOutOfHeapMemory(){ ?
???List?list?=?new?ArrayList();?
???while(true){ ?
??????StringBuffer?B?=?new?StringBuffer();?
??????for(int?i?=?0?;?i?10000?;?i++){ ?
?????????B.append(i);?
??????}?
??????list.add(B);?
???}?
}?
不斷往堆中塞新增的StringBuffer對象,堆滿了就直接溢出了。
特別推薦一個分享架構+算法的優質內容,還沒關注的小伙伴,可以長按關注一下:
長按訂閱更多精彩▼
如有收獲,點個在看,誠摯感謝
免責聲明:本文內容由21ic獲得授權后發布,版權歸原作者所有,本平臺僅提供信息存儲服務。文章僅代表作者個人觀點,不代表本平臺立場,如有問題,請聯系我們,謝謝!