文章目录
内存区域
程序计数器
线程私有 ,可以看作是当前线程所执行的字节码的行号指示器,字节码解释器就是通过改变这个计数器的值来选取下一条需要执行的字节码指令。
java虚拟机的多线程是通过线程轮流切换、分配处理器时间的方式来实现的,即任何时刻,一个处理器只会执行一条线程,因此,为了线程切换后能恢复到原本的执行位置,每个线程就需要一个独立的程序计数器。
如果执行的是一个java方法,记录的是正在执行的虚拟机字节码指令的地址,如果是一个本地方法,则计数器的值应为空。
程序计数器是唯一一个没有规定任何OutOfMemoryError情况的区域
java虚拟机栈
线程私有,他的生命周期与线程相同。
虚拟机栈描述的是java方法执行的线程内存模型,每个方法被执行的时候,java虚拟机都会同步创建一个栈帧用于存储局部变量表、操作数栈、动态连接、方法出口等信息,一个方法被调用直至执行完毕的过程,就对应着一个栈帧在虚拟机栈从入栈到出栈的过程
"栈"一般指的就是虚拟机栈
局部变量表存放了编译器可知的各种Java虚拟机基本数据类型、对象引用类型、returnAddress类型(指向一条字节码指令的地址)这些数据类型在局部变量表中的存储空间以局部变量槽表示,其中64位的long和double会占用两个变量槽,其余数据类型只会占用一个。
局部变量表所需的内存空间在编译期间完成分配,运行期间不会改变局部变量表的“大小”,这里的大小指的是槽的数量。而虚拟机使用多大的内存空间来实现一个变量槽则取决于不同虚拟机的实现。
基本类型 大小 取值范围 装箱基本类型
int 4个字节 -2^31 ~ 2^31-1 Integer
char 2个字节 Character
byte 1个字节 -2^7 ~ 2^7-1 Byte
short 2个字节 -2^15 ~ 2^15-1 Short
long 8个字节 -2^63 ~ 2^63-1 Long
float 4个字节 Float
double 8个字节 Double
boolean 1、4个字节 true ~ false Boolean
当线程请求的栈深度大于虚拟机允许的深度时,抛出StackOverflowError
如果java虚拟机栈容量可以动态扩展,当栈扩展时无法申请到足够的内存空间时会抛出OutOfMemoryError
本地方法栈
线程私有,与虚拟机栈发挥的作用是相似的,区别只是本地方法栈为所使用到的本地方法服务
本地方法栈使用的语言、方式、数据结构取决于不同的虚拟机,甚至有的虚拟机(例如HotSpot)直接将本地方法栈与虚拟机栈合二为一。
当线程请求的栈深度大于虚拟机允许的深度时,抛出StackOverflowError
如果java虚拟机栈容量可以动态扩展,当栈扩展时无法申请到足够的内存空间时会抛出OutOfMemoryError
java堆
线程共享。java Heap是虚拟机所管理的内存中最大的一块,在虚拟机启动时创建,此内存的唯一目的就是存放对象实例。几乎所有的对象实例都在这里分配内存。
java堆是垃圾收集器管理的内存区域。
java 堆可以处于物理上不连续的内存空间中,但是在逻辑上他应该被视为连续的。
java堆可以是固定或者可扩展的,但是当前主流的java虚拟机都是按照可扩展来实现的,通过参数-Xmx和-Xms来设定,当堆没有内存完成实例分配以及无法再扩展时,将抛出OutOfMemoryError
方法区
线程共享,用于存储已被虚拟机加载的类型信息、常量、静态常量、即时编译器编译后的代码缓存等数据,java虚拟机规范把方法区描述为堆的一个逻辑部分,但是他还有一个别名叫做“non heap”,目的就是与java堆区分开来。
方法区与永久代,实际上两者并不等价,因为仅仅是以前的HotSpot虚拟机使用永久代来实现方法区而已,这样的好处是GC能够像管理堆一样管理方法区,但是这种设计会导致了java应用更容易遇到内存溢出的问题。于是在jdk7时,HotSpot把原本放在永久代的字符串常量池、静态变量等移出。jdk8时把在永久代的剩余内容,主要是类型信息全部移到了元空间(Meta
space)中。
java虚拟机规范对方法区的规范是非常宽松的,除了和java堆一样不需要连续的内存和可以选择固定大小或者可扩展外,甚至还可以选择实现垃圾收集,因为垃圾收集行为在这个区域是比较少出现的,这区域的回收目标主要是针对常量池的回收和对类型的卸载,一般来收这个区域的回收效果并不令人满意,尤其是类型卸载,条件相当苛刻,但是这部分的回收有时又确实是有必要的,因为也可能导致内存泄漏
全局字符串池(string pool也有叫做string literal pool)
由上一条可知,在JDK7.0版本,字符串常量池被移到堆中。
全局字符串池里的内容是在类加载完成,经过验证,准备阶段之后在堆中生成字符串对象实例,然后将该字符串对象实例的引用值存到string pool中(记住:string pool中存的是引用值而不是具体的实例对象,具体的实例对