1、程序计数器
程序计数器是当前线程所执行的行号指示器。通过改变计数器的值来确定下一条指令,比如循环,分支,跳转,异常处理,线程恢复等都依赖计数器来完成。
由于在JVM中,多线程是通过线程轮流切换来获得CPU执行时间的,因此在任一时刻,一个CPU的内核只会执行一条线程中的指令,因此,为了能够使得每个线程都在线程切换后能够恢复在切换之前的程序执行位置,每个线程都需要有自己独立的程序计数器,并且不能互相被干扰,否则就会影响到程序的正常执行次序。因此每个计数器是每个线程私有的。
在JVM规范中规定,如果线程执行的是非native方法,则程序计数器中保存的是当前需要执行的指令的地址;如果线程执行的是native方法,则程序计数器中的值是undefind。
由于程序计数器中存储的数据所占空间的大小不会随程序的执行发生改变,因此,程序计数器是不会发生内存溢出(OutOfMemory)现象的。
2、虚拟机栈
虚拟机栈也就是我们所说的栈内存,是java方法执行的内存模型。每个方法在执行的时候都会创建一个栈帧,用于存储局部变量表、操作数栈、动态链接、和方法返回地址等信息。
局部变量表存储的是基本数据类型、returnAdress类型和对象引用。局部变量表的大小在编译期间完成分配,因此程序执行期间局部变量表的大小不会改变。
操作数栈主要用来存储运算结果及运算的操作数。在数据结构中,栈最典型的一个应用就是用来对表达式求值。程序中的所有计算过程都是借助操作数栈完成的。
每个栈帧都包含一个指向运行时常量池中该栈帧所属方法的引用,持有这个引用是为了支持方法调用中的动态链接就是将常量池中的符号引用在运行期间转化为直接引用。
3、本地方法栈:
本地方法栈和虚拟机栈类似,只不过本地方法栈为虚拟机使用本地方法(native)服务。
4、堆
java堆是所有线程共享的一块内存,在虚拟机启动是创建,几乎所有的对象实例都在这里创建,因此该区域经常发生垃圾回收。从内存回收的角度看,由于现在收集器基本都是采用分代收集算法,所以java堆中还可以细分为:新生代和老年代;新生代又分为Eden空间、From Survivor空间、To Survivor空间三部分。
5、方法区
方法区和java堆一样,是各个线程共享的内存区域,不需要连续的内存,并且可以动态扩展,动态扩展失败一样会抛出OutOfMemoryError异常。方法区用于存放已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据。虽然Java虚拟机规范把方法区描述为堆的一个逻辑部分,但是为了与java堆区分开,方法区又叫非堆(Non-Heap),很多人更愿意把方法区称为“永久代”。从jdk1.7已经开始准备“去永久代”的规划,jdk1.7的HotSpot中,已经把原来放在方法区中的静态变量,字符串常量池等移到堆内存中。在jdk1.8中,永久代已经不存在,存储的类信息、即时编译器编译后的代码等已经移动到元空间(MetaSpace)中,元空间并没有处于堆内存上,而是直接占用本地内存(NativeMemory)。
运行时常量池属于方法区的一部分,常量池中存放2类常量:字面量和符号引用。字面量比较接近java语言层面的常量概念,如文本字符串,被声明为final的常量值等。而符号引用则属于编译原理方面的概念,包括3类常量:类和接口的全限定名、字段的名称和描述符,方法的名称和描述符。