一、java.lang.OutOfMemoryError:Java heap space
原因:堆空间不足
1.堆空间设置过小
类所需的内存大,但是设置的jvm参数过小
2.用户数量过大,到达峰值
3.长期的内存泄露
因为代码中没有覆盖equals方法,导致类不断被放到hashmap中,知道将堆空间占满
import java.util.*;
class OOM {
static class Key {
Integer id;
Key(Integer id) {
this.id = id;
}
@Override
public int hashCode() {
return id.hashCode();
}
}
public static void main(String[] args) {
Map<Key,String> m = new HashMap<Key,String>();
while(true) {
for(int i=0;i<10000;i++) {
if(!m.containsKey(new Key(i))) {
m.put(new Key(i), "Number:" + i);
}
}
}
}
}
?对于这种问题,只需要实现equals方法即可
二.java.lang.OutOfMemoryError:GC overhead limit exceeded(gc开销超出限制)
?当应用程序花费超过98%的时间用来做GC并且回收了不到2%的堆内存时,会抛出java.lang.OutOfMemoryError:GC overhead limit exceeded 错误。如果堆内存太小的时候会报这个错误.
三.java.lang.OutOfMemoryError:Permgen space (jdk1.8已取消,用Metaspace代替)
1)初始化时分配的空间不足,可以通过以下设置修改
-XX:MaxPermSize=512m
?2)重部署多次后出现,可以分析dump文件,给应用程序一个关闭的hook
jmap -dump:format=b,file=dump.hprof <process-id>
3)运行时的 OutOfMemoryError,可以先进行如下设置,如果仍然报异常,则需要分析dump文件
-XX:+CMSClassUnloadingEnabled
-XX:+UseConcMarkSweepGC
?四、java.lang.OutOfMemoryError:Metaspace
注:Metaspace所占用的内存空间不是虚拟机内部的,而是本地内存空间。因此,默认情况下,元空间的大小仅受本地内存限制,但可以通过以下参数来指定元空间的大小.
不断的生成新类,类的定义信息被加载到Metaspace,可以通过设置此参数 或者删除此参数
-XX:MaxMetaspaceSize = 512m
五.java.lang.OutOfMemoryError:Unable to create new native thread (栈溢出)
OS设置线程数 ;检查代码;线上程序需要用jstack命令,将当前线程的状态导出来放到文件里边,然后将文件上传到fastthread.io网站上进行分析。
六.java.lang.OutOfMemoryError:Out of swap space
linux增加交换空间
swapoff -a
dd if=/dev/zero of=swapfile bs=1024 count=655360
mkswap swapfile
swapon swapfile
七、java.lang.OutOfMemoryError:Requested array size exceeds VM limit
- 数组增长太大,最终大小在平台限制和
Integer.MAX_INT 之间 - 分配大于
2 ^ 31-1 个元素的数组,编译时就会报错
八、Out of memory:Kill process or sacrifice child ?
当可用虚拟虚拟内存(包括交换空间)消耗到让整个操作系统面临风险时,就会产生Out of memory:Kill process or sacrifice child错误。
解决办法为:升级内存;调整kern的kill配置;内存负载分配到若干实例中
九、 java.lang.StackOverflowError
通过-Xss设置每个线程的堆栈大小,设置这个值越小,能创建更多的线程
十、OOM-Direct buffer memory
public static void main(String[] args) {
final int _1M = 1024 * 1024 * 1;
List<ByteBuffer> buffers = new ArrayList<>();
int count = 1;
while (true) {
ByteBuffer byteBuffer = ByteBuffer.allocateDirect(_1M);
buffers.add(byteBuffer);
System.out.println(count++);
}
}
?
?
|