JVM学习4·内存溢出

内存溢出,也就是Out Of Memory,应该是比较常见的问题吧,这篇我们就好好分析一下内存溢出的几种情况

1.栈溢出 (java.lang.StackOverflowError)

1.1方法自调用例子

栈溢出就是发生在栈的内存空间溢出的情况,以下这个例子就很好的可以模拟出这样的环境

image

图中方法就是不断的自己调用自己,这样导致就是不断的将方法压栈,但是无法出栈,这样就导致本来就有单个虚拟机栈内存限制(默认1M)的栈空间终将溢出。

1.2线程过多

关于虚拟机栈整体是没有内存限制的,但是对于单个的虚拟机栈是有限制,默认是1M,那么如果线程一直不断的新增启动就会导致栈空间终将耗尽,最终导致整个机器卡死。

2.堆溢出

2.1 对象过多 java.lang.OutOfMemoryError: Java heap space

如图,我们在jvm设置了堆空间的大小为30M,结果一下子就创建了超过30M的数组,当然一下就直接溢出了。

image

2.2 GC占据98%的资源 GC Overhead Limit Exceeded Error

image

如图,我们创建了一个集合,然后不断的往里面添加对象,那么最终会达到堆内存的限制,在限制之前会不断的GC回收垃圾,但是每次都无法回收很多垃圾,因为最多的对象仍然被集合引用着,所以说有很多的GC,但是就是无法回收垃圾,最终导致内存溢出。

OutOfMemoryError是java.lang.VirtualMachineError的子类,当JVM资源利用出现问题时抛出,更具体地说,这个错误是由于JVM花费太长时间执行GC且只能回收很少的堆内存时抛出的。根据Oracle官方文档,默认情况下,如果Java进程花费98%以上的时间执行GC,并且每次只有不到2%的堆被恢复,则JVM抛出此错误。换句话说,这意味着我们的应用程序几乎耗尽了所有可用内存,垃圾收集器花了太长时间试图清理它,并多次失败。

在这种情况下,用户会体验到应用程序响应非常缓慢,通常只需要几毫秒就能完成的某些操作,此时则需要更长的时间来完成,这是因为所有的CPU正在进行垃圾收集,因此无法执行其他任务。在此我向大家推荐一个架构学习交流圈。交流学习指导伪鑫:1253431195(里面有大量的面试题及答案)里面会分享一些资深架构师录制的视频录像:有Spring,MyBatis,Netty源码分析,高并发、高性能、分布式、微服务架构的原理,JVM性能优化、分布式架构等这些成为架构师必备的知识体系。还能领取免费的学习资源,目前受益良多

扫描二维码关注公众号,回复: 16174695 查看本文章

3.方法区溢出

在之前的JVM内存结构中我们知道在方法区会存放静态常量,静态变量,还有类信息等。所以如果想导致方法区内存溢出,我们就可以不断的添加类来导致:

首先我们需要添加cglib这个包:

image

然后就可以使用该包提供的方法不断的加载类:

还要设置一下方法区的大小限制来模拟

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-ea1fidgC-1652354857463)(https://upload-images.jianshu.io/upload_images/8553141-1d2c0977b8301ef7.png)]

image

image

4.本机直接内存溢出

这块我们基本不会用到,他就是操作在jvm之外的内存
image

猜你喜欢

转载自blog.csdn.net/muli522/article/details/124737969