摘自kakaluyi的<如何定位OutOfMemory的根本原因>,有部分修改
java的OutOfMemory有关Exception和可能出现的方式:
A Exception in thread "main" java.lang.OutOfMemoryError: PermGen space
java的OutOfMemory有关Exception和可能出现的方式:
A Exception in thread "main" java.lang.OutOfMemoryError: PermGen space
- public static void main(String[] args) {
- //使用List保持着常量池引用,压制Full GC回收常量池行为
- List<String> list = new ArrayList<String>();
- // 10M的PermSize在integer范围内足够产生OOM了
- int i = 0;
- while (true) {
- list.add(String.valueOf(i++).intern());
- }
- }
public static void main(String[] args) {
//使用List保持着常量池引用,压制Full GC回收常量池行为
List<String> list = new ArrayList<String>();
// 10M的PermSize在integer范围内足够产生OOM了
int i = 0;
while (true) {
list.add(String.valueOf(i++).intern());
}
}
永久区溢出
这一部分用于存放Class和Meta的信息,Class在被 Load的时候被放入PermGen space区域(包括常量池: 静态变量),它和存放Instance的Heap区域不同,GC(Garbage Collection)不会在主程序运行期对PermGen space进行清理,所以如果你的APP会LOAD很多CLASS的话,就很可能出现PermGen space错误。
B java.lang.OutOfMemoryError: Java heap space
- public static void main(String[] args) {
- //使用List保持着常量池引用,压制Full GC回收常量池行为
- List<String> list = new ArrayList<String>();
- // 10M的PermSize在integer范围内足够产生OOM了
- int i = 0;
- while (true) {
- list.add(new String(i++));
- }
- }
public static void main(String[] args) {
//使用List保持着常量池引用,压制Full GC回收常量池行为
List<String> list = new ArrayList<String>();
// 10M的PermSize在integer范围内足够产生OOM了
int i = 0;
while (true) {
list.add(new String(i++));
}
}
堆溢出
这部分用于存放类的实例。被缓存的实例(Cache)对象,大的map,list引用大的对象等等,都会保存于此。
C Exception in thread "main" java.lang.StackOverflowError
- public class JavaVMStackSOF {
- private int stackLength = 1;
- public void stackLeak() {
- stackLength++;
- stackLeak();
- }
- public static void main(String[] args) throws Throwable {
- JavaVMStackSOF oom = new JavaVMStackSOF();
- try {
- oom.stackLeak();
- } catch (Throwable e) {
- System.out.println("stack length:" + oom.stackLength);
- throw e;
- }
- }
- }
public class JavaVMStackSOF {
private int stackLength = 1;
public void stackLeak() {
stackLength++;
stackLeak();
}
public static void main(String[] args) throws Throwable {
JavaVMStackSOF oom = new JavaVMStackSOF();
try {
oom.stackLeak();
} catch (Throwable e) {
System.out.println("stack length:" + oom.stackLength);
throw e;
}
}
}
栈溢出
这部分用于存放局部变量、方法栈帧信息。栈帧太多,也就是函数调用层级过多时就会出现此异常,检查是否有死递归的情况。
今天碰到个内存溢出的问题,解决过程如下:
- 首先启动时添加参数-XX:+HeapDumpOnOutOfMemoryError,这样当内存溢出时:
- java.lang.OutOfMemoryError: Java heap space
- Dumping heap to java_pid5304.hprof ...
- Heap dump file created [85357895 bytes in 2.095 secs]
- 会在eclipse目录下生产内存文件;
- 使用Memory Analyze分析,查看类使用情况,发现是由Map过大引起;
- 查看代码中使用到Map的段,分析,解决问题;
问题over。