Java运行时内存区域

2020-05-10  本文已影响0人  ACtong

一、运行时内存区域

Java虚拟机在执行Java程序的过程,会把它所管理的内存划分为若干个不同的数据区域


Java虚拟机运行时数据区

二、程序计数器(Program Counter Register)

程序计数器:记录当前方法的虚拟机字节码的指令地址。你可以把它看成当前线程中的所执行的字节码行号指示器。

在Java虚拟机的概念模型里,字节码解释器是,它工作时,通过改变程序计数器的值来选取下一条需要执行的字节码指令。它是程序控制流的指示器,分支,循环、跳转、异常处理、线程恢复等功能都需要依赖这个程序计数器。

在多线程中,每个线程都需要一个独立的程序计数器,各个线程之间的计数器互不影响,独立存储,我们称这类内存区域为“线程私有”的内存。

内存溢出:

此内存区域是唯一一个在《Java虚拟机规范》中没有规定任何OutOfMemoryError情况的区域

三、虚拟机栈(VM Stack)

当一个方法被执行时,虚拟机会同步创建一个栈帧(Stack Frame),用来存储局部变量表、操作数栈、动态连接、方法出口等信息,每一个方法创建到执行完毕的过程,就对应着栈帧在虚拟机中入栈和出站的过程。这就是虚拟机栈描述的Java方法执行的线程内存模型。

Java虚拟机栈:线程私有,生命周期和线程相同。
通常我们所说的栈,就是虚拟机栈,或者更多时候是指虚拟机栈中的局部变量表部分

局部变量表存放:基本数据类型(byte,short,int,long,float,double,char,boolean),引用类型,returnAddress类型(指向一条字节码指令的地址)

内存溢出:
  1. 如果线程请求的栈深读大于虚拟机所允许的深度,将抛出StackOverflowError
  2. 如果虚拟机栈容量可以动态扩展, 当扩展时无法申请到足够内存会抛出OutOfMemoryError异常(HotSpot虚拟机的栈容量是不可以动态扩展的)

四、本地方法栈(Native Method Stack)

本地方法栈与虚拟机栈相似,唯一的区别就是:

五、Java堆

Java堆:唯一目的就是存放对象实例,Java 世界里“几乎”所有的对象实例都在这里分配内存(几乎是指的现在,不代表将来会出现值类型的支持)

如果从内存分配的角度来看,所有线程共享的Java堆中可以划分出多个线程私有的分配缓冲区(Thread Local Allocation Buffer,TLAB),以提升对象分配时的效率。

但是无论从什么角度,无论如何划分,都不会改变Java堆中存储内容的共性,无论哪个区域,存储的都只能时对象实例,将Java堆细分,仅仅是为了更好的内存回收,或者更快的分配内存。

内存溢出:

六、方法区

方法区(Method Area):与Java堆一样,是各个线程共享的内存区域,它用于存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码缓存等数据。
栈:存储局部变量表、操作数栈、动态连接、方法出口等信息。

方法区内存回收目标:主要是针对常量池的回收和对类型的卸载,一般来说回收效果难以令人满意,卸载效果更是苛刻。

内存溢出:

方法区无法满足新的内存分配需求时,将抛出OutOfMemoryError异常。

七、运行时常量池

内存溢出:

运行时常量池是方法区的一部分,自然受到方法区内存的限制,当常量池无法再申请到内存 时会抛出OutOfMemoryError异常。

上一篇 下一篇

猜你喜欢

热点阅读