内存溢出,也就是Out Of Memory,应该是比较常见的问题吧,这篇我们就好好分析一下内存溢出的几种情况
1.栈溢出 (java.lang.StackOverflowError)
1.1方法自调用例子
栈溢出就是发生在栈的内存空间溢出的情况,以下这个例子就很好的可以模拟出这样的环境
图中方法就是不断的自己调用自己,这样导致就是不断的将方法压栈,但是无法出栈,这样就导致本来就有单个虚拟机栈内存限制(默认1M)的栈空间终将溢出。
1.2线程过多
关于虚拟机栈整体是没有内存限制的,但是对于单个的虚拟机栈是有限制,默认是1M,那么如果线程一直不断的新增启动就会导致栈空间终将耗尽,最终导致整个机器卡死。
2.堆溢出
2.1 对象过多 java.lang.OutOfMemoryError: Java heap space
如图,我们在jvm设置了堆空间的大小为30M,结果一下子就创建了超过30M的数组,当然一下就直接溢出了。
2.2 GC占据98%的资源 GC Overhead Limit Exceeded Error
如图,我们创建了一个集合,然后不断的往里面添加对象,那么最终会达到堆内存的限制,在限制之前会不断的GC回收垃圾,但是每次都无法回收很多垃圾,因为最多的对象仍然被集合引用着,所以说有很多的GC,但是就是无法回收垃圾,最终导致内存溢出。
OutOfMemoryError是java.lang.VirtualMachineError的子类,当JVM资源利用出现问题时抛出,更具体地说,这个错误是由于JVM花费太长时间执行GC且只能回收很少的堆内存时抛出的。根据Oracle官方文档,默认情况下,如果Java进程花费98%以上的时间执行GC,并且每次只有不到2%的堆被恢复,则JVM抛出此错误。换句话说,这意味着我们的应用程序几乎耗尽了所有可用内存,垃圾收集器花了太长时间试图清理它,并多次失败。
在这种情况下,用户会体验到应用程序响应非常缓慢,通常只需要几毫秒就能完成的某些操作,此时则需要更长的时间来完成,这是因为所有的CPU正在进行垃圾收集,因此无法执行其他任务。
3.方法区溢出
在之前的JVM内存结构中我们知道在方法区会存放静态常量,静态变量,还有类信息等。所以如果想导致方法区内存溢出,我们就可以不断的添加类来导致:
首先我们需要添加cglib这个包:
然后就可以使用该包提供的方法不断的加载类:
还要设置一下方法区的大小限制来模拟
4.本机直接内存溢出
这块我们基本不会用到,他就是操作在jvm之外的内存