开门见山。
为了方便制造溢出,将JAVA堆的大小调整为10M。
本机用的是IntelliJ IDEA作为开发工具,进入到IDEA的安装目录,如D:\tools\IntelliJ IDEA 2017.1.4\bin,打开idea64.exe.vmoptions文件(32位的请打开idea.exe.vmoptions),把-Xms(最大堆内存)和-Xmx(初始化对内存大小)都设置成10M,书上说设置成一样可避免堆自动扩展(测试了一下,设置成不一样并没有扩展啊),然后重启IDEA。
1 堆溢出
前面已经介绍过了,堆是用来存储对象实例的。如果对象达到一定的数量,并且这些对象和根对象之间有可达路径,那么对象就不能被垃圾收集器回收,超过堆的最大容量,就会出现溢出。
代码如下,将一个Integer数组,不停地添加到list集合中,当循环到88800时,程序报错:Exception in thread "main" java.lang.OutOfMemoryError: Java heap space。
对于这个错误,有过开发经验的朋友肯定很熟悉,实际开发中,很可能是因为逻辑错误而出现死循环造成的。
当然,还有另外一种情况,那就是这种持有对象是必要的,那么就要考虑机器物理内存的大小,来适当的调整虚拟机堆的大小,同时优化代码。
2 虚拟机栈和本地方法栈溢出
关于虚拟机栈和本地方法栈,Java虚拟机描述了*Error和OutOfMemoryError两种异常。
相信有部分兄弟对这两个错误分的不太清晰,包括我也是,先简单说一下这两种异常的区别。
*Error:字面意思-栈溢出错误,那很明显,这个错误是和栈有关系的,什么时候会出现这个错误呢?线程请求的栈深度大于虚拟机所允许的最大深度!
OutOfMemoryError:字面意思-内存溢出错误,书中简称OOM(并不喜欢这个简称,感觉不直观),这个错误很明显是和内存有关系的,什么时候出现这个错误呢?虚拟机在扩展时无法申请到足够的内存空间!
书中说“其本质上是对同一件事情的两种描述”,笔者认为可以换一种理解方式。
比如你拿了1000块钱要去买手机,看重了一个1500的,店员说:“不好意思,您只能买1000元以内的”——栈溢出了。
然后你去找你媳妇,说能不能给我500块钱,我要买手机,你媳妇说:“老娘买化妆品还没钱呢”——内存溢出了。
哈哈,开个玩笑。
下面运行如下包含递归方法的程序。
当运行到24134次时,程序报错了,Exception in thread "main" java.lang.*Error。这个24134就是所说的栈的最大深度了。
这里有一点需要注意!注意!注意!当涉及到多线程的时候,并不是为每个线程的栈分配的内存越大越好,因为操作系统分配给JVM虚拟机的内存也是有限的,搞不好会产生内存溢出的。
3 方法区和运行时常量溢出
复习一下,方法区用于存放Class的相关信息,如类名,访问修饰符,常量池,静态变量,字段描述,方法描述,即时编译后的代码等数据。
如果运行时方法区产生了大量的类,超出了方法区的最大容量,就会产生内存溢出。
注:jdk8中永久区被移除了,取而代之的是元数据区,元数据区是堆外直接内存,可以提供更大的空间。
4 本机直接内存溢出
本机直接内存溢出相对比较少见,这里简单介绍一下,主要是当直接向操作系统分配内存的时候,内存无法分配的时候,手动抛出异常。
喜欢文章或想一起学习的朋友可以关注我,我将会持续更新,有什么疑问或文中有不当之处请给我留言,真诚地希望能与大家一起交流探讨,学习进步。