JVM的内存结构
根据 JVM 规范,JVM 内存共分为虚拟机栈、堆、方法区、程序计数器、本地方法栈五个部分。
各区域的作用:
OOM
OOM,全称
Out Of Memory
,当JVM因为没有足够的内存来为对象分配空间,并且垃圾回收器也已经没有空间可回收时,就会抛出这个error
。
- 常见的内存溢出现象:堆内存溢出
- 在JVM可使用的最大堆内存可以在启动的时候通过
-Xmx
参数指定。堆内存溢出是最为常见的内存溢出问题,发生堆内存溢出时,JVM会报告如下错误:java.lang.OutOfMemoryError: java heap space
。 - 分析:内存溢出顾名思义就是堆内存不够用了。如果程序设计的最大堆内存已经耗尽,那说明程序设计存在问题,不该申请很多内存的逻辑申请了很多内存,该释放的对象没有释放。
- 解决方式:
- 检查程序,看看是否存在死循环或不必要地重复地创建大量对象。找到原因后,修改程序和算法。
- 增加Java虚拟机中
Xms
(初始堆大小)和Xmx
(最大堆大小)参数的大小。如:set JAVA_OPTS= -Xms256m -Xmx1024m
- 在JVM可使用的最大堆内存可以在启动的时候通过
堆与栈的区别
- 堆是运行时确定内存大小,而栈在编译时即可确定内存大小。
- 堆内存由用户管理(Java中由JVM管理),栈内存会被自动释放。
- 栈实现方式采用数据结构中的栈实现,具有先进后出的顺序特点,堆为一块一块的内存。
- 栈由于其实现方式,在分配速度上比堆快的多。分配一块栈内存不过是简单的移动一个指针。
- 栈为线程私有,堆为线程共享。