OOM异常 Java内存溢出
1.OutOfMemoryError
- 抛出异常后先确定是堆溢出还是栈溢出
-
堆溢出:java.lang.OutOfMemoryError: Java heap space
- 堆出现OOM(标志就是Java heap space)首先确认是内存溢出还是内存泄漏。常见的OOM就是堆溢出。
- 内存泄漏: Memory Leak 申请到内存之后无法 释放 申请到的内存,造成这种情况的就是某个对象一直被引用这。
- 内存溢出:Memory OverFlow :申请内存时没有足够的空间可以使用,抛出OOM。
- 可以使用 Eclipse Memory Analyzer 或者 JProfiler(IDEA)分析内存是否泄漏
- -Xms 设置初始堆的大小,设置为20MB -Xms20m
- -Xmx设置最大堆的大小
-
虚拟机栈溢出和本地方法栈溢出
- HotSpot 虚拟机并不区分虚拟机栈和本地方法栈他两适合在一起的所以 用来设置本地方法栈大小的-Xoos并没有什么效果。
- -Xss 设置栈容量大小
- 栈可以抛出2种异常
- StackOverflowError 线程请求的栈深度大于虚拟机所允许的最大深度。
- OutOfMemoryError 扩展栈时无法申请到足够的内存,比较少见。
- 操作系统分配给每个进程的内存是有限的,如32位win给进程分配的是2G,栈内存 = 2GB - 堆Xmx - 方法区 MaxPermSize
- 线程越多分配的栈内存就越少,每个线程分配的栈容量就越大线程就越少,容易把剩下的内存耗尽。
- jvm默认情况下栈深度平均在1000-2000间(栈帧个数即方法数量),如果不能更换更高位数的操作系统也不能减少线程数量,那么可以通过减少堆的最大容量让出一部分内存来给栈,避免栈内存溢出。
-
方法区和运行时常量池溢出
- OutOfMemoryError:PermGen space 运行时常量池在方法区中他两都会抛出此异常。
- -XX:PermSize 方法区初始值大小
- -XX:MaxPermSize 方法区最大值
- 由于1.8 去永久带 而加入了 Metaspace 元空间 所以上面的配置 1.8 开始就不能失效了
- -XX:MetaspaceSize 元空间大小, -XX:MaxMetaspace元空间最大值。
-
本机直接内存溢出
- -XX: MaxDirectMemorySize 可以指定最大值,如果不指定默认和堆一样大。
- 由于直接内存不在堆里,它是供NIO使用的所以抛出OOM异常后通过jvm内存检测工具 在Heap Dump中没有明显错误,但是Dump文件很小就可以考虑直接内存溢出
知识点:
- string.intern()是一个本地方法,它只 返回字符串常量池中唯一的那个string。
String st1 = new StringBuilder("Mi").append("bloom").toString(); // 初始字符串常量池不存在组合的Mibloom ,但存在了Mi和bloom. String st2 = new StringBuilder("ja").append("va").toString();// 初始字符串常量池存在java System.out.println(st1.intern() == st1);// 1.6为 false,1.7为true System.out.println(st2.intern() == st2);//始终为false, 常量池中存在java,因此 st2.intern()返回的是常量池中的,而st2是堆中的。
- 只要代码中直接出现的字符串就还会被放入常量池,但是组合生成的字符串不会被放入常量池的,如常量池中有Mi和bloom,但是没有Mibloom。
- 如果字符串常量池中已经包含了一个等于此string对象的符号,则string.intern()返回字符串常量池中的对象。
- 如果字符串常量池中没有,则将string加入字符串常量池并返回字符串常量池中的对象,此加入操作在jdk1.6 和1.7 中有所不同。
- 1.6中首次出现的string加入字符串常量池中的是直接复制到永久代中的字符串常量池中,所以地址会发生变化。
- 1.7开始 首次出现的string加入字符串常量池是复制string的引用到常量池中,所以地址不会发生变化。1.7开始 字符串常量池被移到了堆中。
- JVM 其他配置参数
//常见配置汇总 //堆设置 -Xms:初始堆大小 -Xmx:最大堆大小 -XX:NewSize=n:设置年轻代大小 -XX:NewRatio=n:设置年轻代和年老代的比值.如:为3,表示年轻代与年老代比值为1:3,年轻代占整个年轻代年老代和的1/4 -XX:SurvivorRatio=n:年轻代中Eden区与两个Survivor区的比值.注意Survivor区有两个.如:3,表示Eden:Survivor=3:2,一个Survivor区占整个年轻代的1/5 -XX:MaxPermSize=n:设置持久代大小 //收集器设置 -XX:+UseSerialGC:设置串行收集器 -XX:+UseParallelGC:设置并行收集器 -XX:+UseParalledlOldGC:设置并行年老代收集器 -XX:+UseConcMarkSweepGC:设置并发收集器 //垃圾回收统计信息 -XX:+PrintGC -XX:+PrintGCDetails -XX:+PrintGCTimeStamps -Xloggc:filename //并行收集器设置 -XX:ParallelGCThreads=n:设置并行收集器收集时使用的CPU数.并行收集//线程数. -XX:MaxGCPauseMillis=n:设置并行收集最大暂停时间 -XX:GCTimeRatio=n:设置垃圾回收时间占程序运行时间的百分比.公式为1/(1+n) //并发收集器设置 -XX:+CMSIncrementalMode:设置为增量模式.适用于单CPU情况. -XX:ParallelGCThreads=n:设置并发收集器年轻代收集方式为并行收集时,使用的CPU数.并行收集线程数.