Java虚拟机在执行Java程序的过程中会把它所管理的内存划分为若干个不同的数据区域。
![](https://i-blog.csdnimg.cn/blog_migrate/7c569498e83165c45995981aeee36e2c.png#pic_center)
1.程序计数器:可以看作是当前线程所执行的字节码行号指示器。(线程私有)
由于Java虚拟机的多线程是通过线程轮流切换、分配处理器执行时间的方式来实现的,在任何一个确定的时刻,一个处理器都会执行一条线程中的指令。为了保证线程切换后能回到原来的正确位置,每条线程都需要有有一个独立的程序计数器;各条程序计数器独立存储互不影响,这类内存区域称之为“线程私有”的内存。
如果线程正在执行的是Java方法,那么这个计数器记录的是正在执行的虚拟机字节码指令的地址;如果线程执行的是本地方法,这个计数器的值为空(Undefined)
这个区域没有规定任何OutOfMemoryError情况
2.Java虚拟机栈:描述Java方法执行的线程内存模型。每个方法被执行的时候,Java虚拟机都会同步创建一个栈帧用于存储局部变量表、操作数栈、动态连接、方法出口等信息;每一个方法被调用直至执行完毕的过程,就对应一个栈帧在虚拟机栈中从入栈到出栈的过程。(线程私有,生命周期和线程同步)
这个内存区域规定了两类异常状况:如果线程请求的栈深度大于虚拟机所允许的深度,将抛出StackOverflowError异常;如果Java虚拟机栈容量可以动态扩展,当栈扩展时无法申请到足够的内存会抛出OutOfMemoryError异常。
3.本地方法栈:为虚拟机使用到的本地方法服务,和Java虚拟机栈类似。(线程私有)
本地方法栈也会在栈深度溢出或者栈扩展失败时分别抛出StackOverflowError和OutOfMemoryError异常。
4.Java堆:存放对象实例,在虚拟机启动时创建,是虚拟机所管理内存中最大的一块。(线程共享)
Java堆是垃圾收集器管理的内存区域。
Java堆可以被实现固定大小,也可以是可扩展的。当前主流的Java虚拟机都是按照可扩展来实现的(通过参数-Xmx和-Xms设定)。如果在Java堆里没有内存完成实例分配,且堆也无法再扩展时,Java虚拟机将会抛出OutOfMemoryError异常。
5.方法区:用于存储已被虚拟机加载的类的信息、常量、静态变量、即时编译器编译后的代码缓存等数据。(线程共享)
*运行时常量池:为方法区的一部分,用于存放编译期生成的各种字面量与符号引用