您好,登錄后才能下訂單哦!
如今不管是在面試還是在我們的工作中,OOM總是不斷的出現在我們的視野中,所以我們有必要去了解一下導致OOM的原因以及一些基本的調整方法,大家可以通過下面的事例來了解一下什么樣的代碼會導致OOM,幫助我們以后在工作中能夠通過異常信息來判斷是JVM里面哪個區域出現了問題。
先介紹一下筆者的相關編碼環境。
jdk:java version "1.8.0_121"
ide:IntelliJ IDEA 2019.1 (Community Edition)
Java中的堆存儲的都是對象實例,當我們不斷的創建對象,而GC的時候又不能回收,當存儲的對象大小超過了-Xmx的值,這時候則會出現OutOfMemoryError.[-XX:+HeapDumpOnOutOfMemoryError]參數可以讓jvm出現內存溢出的時候dump出內存堆轉儲快照。
/**
* VM Args: -Xms10m -Xmx10m -XX:+HeapDumpOnOutOfMemoryError
* @author wangzenghuang
*/
public class HeapOOMDemo {
public static void main(String[] args) {
List<String> stringList = new ArrayList<>();
while(true){
stringList.add("str");
}
}
}
運行結果,發生OOM,并且在我們項目的根目錄dump出當前的內存堆快照
java.lang.OutOfMemoryError: Java heap space
Dumping heap to java_pid1376.hprof ...
Heap dump file created [7972183 bytes in 0.047 secs]
Exception in thread "main" java.lang.OutOfMemoryError: Java heap space
at java.util.Arrays.copyOf(Arrays.java:3210)
at java.util.Arrays.copyOf(Arrays.java:3181)
at java.util.ArrayList.grow(ArrayList.java:261)
at java.util.ArrayList.ensureExplicitCapacity(ArrayList.java:235)
at java.util.ArrayList.ensureCapacityInternal(ArrayList.java:227)
at java.util.ArrayList.add(ArrayList.java:458)
at HeapOOMDemo.main(HeapOOMDemo.java:12)
Process finished with exit code 1
簡單解決思路
那么發生這個問題以后我們的解決思路有哪些呢?我們可以利用一些工具(例如Eclipse Memory Analyzer
)來分析dump出的文件,一般來說,當生產環境發生OOM,比較常見的一個原因是發生了內存泄漏,用工具可以分析出泄露的對象到GC Root的引用鏈,從而定位到問題代碼。假如經過分析后發現內存中的對象都是“必須存活”的對象,這時候就要思考下項目中是否把“-Xms跟-Xmx”設置得太小了(當然這里也不是隨意調大,需要結合機器的物理內存情況),再者需要留意代碼中是否有一些長生命周期的對象,從代碼中優化內存消耗。
在jvm的方法區中,它主要存放了類的信息,常量,靜態變量等。在jdk8以前是通過“-XX:PermSize,-XX:MaxPermSize”來調整這個區域的值,但是從8開始呢,永久代的概念被MetaSpace(元空間)代替了,對應的參數也變成了“-XX:MetaspaceSize,-XX:MaxMetaspaceSize”。在這個例子中使用CGLib來動態生成一些類,方便我們實驗操作。
/**
* VM Args: -XX:MetaspaceSize=5m -XX:MaxMetaspaceSize=5m
* @author wangzenghuang
*/
public class MethodAreaOOMDemo {
public static void main(String[] args) {
while(true){
Enhancer enhancer = new Enhancer();
enhancer.setSuperclass(OOMObject.class);
enhancer.setUseCache(false);
enhancer.setCallback(new MethodInterceptor() {
public Object intercept(Object obj, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
return methodProxy.invokeSuper(obj,objects);
}
});
enhancer.create();
}
}
static class OOMObject{}
}
運行結果
Exception in thread "main" java.lang.OutOfMemoryError: Metaspace
簡單解決方法
這個問題的話,一般來說根據情況調整方法區的大小就行了,網上也有人說可以去掉MetaSpace的的大小限制,但是不建議這么干,畢竟不可控的事情我們要少點干,很容易給自己埋雷。
對于我們來說,還有一個熟悉的錯誤,那就是“StackOverflowError”,它是由線程請求的棧深度超過了jvm允許的最大范圍而產生的。“-Xss”參數可以設置棧容量。
/**
* VM Args: -Xss128k
* @author wangzenghuang
*/
public class StackOFDemo {
private static int stackLength = 1;
public void stackLeak(){
stackLength++;
stackLeak();
}
public static void main(String[] args) {
StackOFDemo stackOFDemo = new StackOFDemo();
try {
stackOFDemo.stackLeak();
}catch (Throwable e){
System.out.println("length : "+ stackLength);
throw e;
}
}
}
運行結果
length : 983
Exception in thread "main" java.lang.StackOverflowError
at StackOFDemo.stackLeak(StackOFDemo.java:10)
at StackOFDemo.stackLeak(StackOFDemo.java:10)
...
簡單解決思路
一般來說此類問題多出現在存在遞歸的地方,要從代碼里重新審視遞歸未結束的原因,若遞歸的方法沒問題可以根據實際情況調整“-Xss”參數的大小。還有一些代碼的循壞依賴也會造成此類情況,
本機直接內存默認與“-Xmx”設定的值一樣大,可以通過“-XX:MaxDirectMemorySize”修改。
/**
* VM Args: -Xmx20m -XX:MaxDirectMemorySize=10
* @author wangzenghuang
*/
public class DirectMemoryOOMDemo {
private static final int _1MB = 1024 * 1024;
public static void main(String[] args) throws IllegalAccessException {
Field field = Unsafe.class.getDeclaredFields()[0];
field.setAccessible(true);
Unsafe unsafe = (Unsafe) field.get(null);
while (true){
unsafe.allocateMemory(_1MB);
}
}
}
運行結果
呃,一運行這段代碼idea直接閃退了,查閱其他資料可以得知當DirectMemory導致內存溢出時,Heap Dump文件是很小的,如果程序中有使用NIO的情況可以檢查一下。
這里所展示的代碼只是可以觸發jvm的各種錯誤,但是并不代表這是唯一的觸發錯誤的方方式,假如我們的代碼比較復雜,有時候遇到類似錯誤的時候還是需要耐心分析。
公眾號博文同步Github倉庫,有興趣的朋友可以幫忙給個Star哦,碼字不易,感謝支持。
https://github.com/PeppaLittlePig/blog-wechat
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。