小编给大家分享一下JVM中内存区域与内存溢出的示例分析,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!
Java内存区域与内存溢出异常
运行时数据区域

程序计数器
当前线程所执行的字节码的行号指示器
当前线程私有
不会出现OutOfMemoryError情况
java虚拟机栈
线程私有,生命周期与线程相同
java方法执行的内存模型,每个方法执行的同时都会创建一个栈帧,存储局部变量表(基本类型、对象引用)、操作数栈、动态链接、方法出口等信息
StackOverflowError异常:当线程请求的栈深度大于虚拟机所允许的深度
OutOfMemoryError异常:如果栈的扩展时无法申请到足够的内存
本地方法栈
与虚拟机栈相似,主要为虚拟机使用到的Native方法服务,在HotSpot虚拟机中直接把本地方法栈与虚拟机栈二合一
Java堆(Java Heap)
java堆是被所有线程共享的一块内存区域,在 虚拟机启动时创建。此区域的***目的就是存储对象实例。java堆是垃圾收集器管理的主要区域。java堆还可以细分为:新生代与老年代。在细一点有 Eden空间、Form Survivor空间、To Survivor空间等。
方法区
运行时常量池
直接内存
NIO可以使用Native函数库直接分配堆外内存,堆中的DirectByteBuffer对象作为这块内存的引用进行操作
大小不受Java堆大小的限制,受本机(服务器)内存限制
OutOfMemoryError异常:系统内存不足时
HotSpot虚拟机
对象的创建
虚拟机遇到一条new指令时,首先将去检查这个对象的参数是否在常量池中定位到一个类的符号引用,并且检查这个符号引用代表的类是否已被加载、解析和初始化过。如果没有,必须先执行类的加载过程。
在类加载检查通过后,虚拟机将为新生对象分配内存。对象所需内存大小再类加载完成后便可确定。内存分配可以采用“指针碰撞”与“空闲列表”的方式。
对象的访问定位
java程序需要通过栈上的reference数据来操作堆上的具体对象。访问方式有使用句柄和直接指针两种。
OOM异常的解决思路
生成Dump快照文件:
通过jvm参数—XX:-HeapDumpOnOutOfMemoryError可以让JVM在出现内存溢出是Dump出当前的内存转储快照
用jmap生产dump文件,win通过任务管理器查看tomcat的进程pid,linux用ps命令查看进程pid,然后用jmap命令
先通过内存映像分析工具(如Eclipse的Memory Analyzer)进行分析,常见的情况有:
OOM异常示例:
package oom;
import java.util.ArrayList; import java.util.List; /** * VM Args: -Xms20m -Xmx20m -XX:+HeapDumpOnOutOfMemoryError * @ClassName: HeapOOM * */ public class HeapOOM { static class OOMObject{ } public static void main(String[] args) { List<OOMObject> list = new ArrayList<OOMObject>(); while(true){ list.add(new OOMObject()); } } }以上是“JVM中内存区域与内存溢出的示例分析”这篇文章的所有内容,感谢各位的阅读!相信大家都有了一定的了解,希望分享的内容对大家有所帮助,如果还想学习更多知识,欢迎关注天达云行业资讯频道!