

程序计数器是一块较小的内存空间,可以看做是当前线程所执行的字节码的行号指示器,各线程之间计数器互不影响。
程序计数器是唯一一个不会出现OutOfMemoryError的内存区域,它的生命周期与线程同步。
除了一些Native方法调用通过本地方法栈实现,其他所有的Java方法调用都是通过栈来实现的,每一次方法调用都会入栈,每一个方法返回都会出栈,每个方法对应一个栈帧,栈帧内部结构如下:

类似于虚拟机栈,虚拟机栈为虚拟机执行Java方法(字节码)服务,本地方法栈为虚拟机使用的Native方法服务,在HotSpot虚拟机中,两栈合二为一。
Native方法被执行时在本地方法栈也会创建栈帧,结构同上。
堆是Java虚拟机所管理的内存中最大的一块,是所有线程共享的一块内存区域,唯一作用是存放对象实例,几乎所有的对象实例以及数组都在这里分配内存。
但随着JIT编译器的发展产生了逃逸分析技术,如果某些方法中的对象引用没有被返回或者未被外面使用,那么对象可以直接在栈上分配内存。
Java堆是垃圾收集器管理的主要区域,在JDK7及之前,堆从垃圾回收的角度被划分为新生代、老年代和永久代;在JDK8之后永久代被元空间取代,元空间使用本地内存。
方法区是一种设计规范,属于JVM运行时数据区域的一块逻辑区域,是各个线程共享的内存区域,当虚拟机要使用一个类时,它需要读取并解析Class文件获取相关信息,再将信息存入方法区,主要是类信息、字段信息(成员变量)、方法信息、常量、静态变量等。
永久代和元空间是实现方法区的两种方式,弃用永久代的主要原因是: 整个永久代有一个JVM本身设定的固定上限,不能调整,而元空间放在本地内存,不容易溢出。
符号引用以一组符号来描述所引用的目标,可以是任何形式的字面量,比如类和接口的全限定名、字段的名称和描述符、方法的名称和描述符等,在编译期或者运行期间生成,不依赖于具体的内存地址,而是在运行时根据上下文信息去定位目标。
直接引用时一种直接指向目标的内存地址或者偏移量,与内存地址直接相关,如指向对象实例的指针、指向类的变量的指针等。
在程序运行时需要通过符号引用来找到对应的直接引用,这个过程称为解析,他是Java虚拟机执行引擎的一部分。
使用两种引用的原因:
常量池表,用于存放编译期生成的各种字面量和符号引用,类似符号表。
字面量是源代码中的固定值,包括整数、浮点数和字符串字面量。
符号引用包括类符号引用、字段符号引用、方法符号引用、接口方法符号等。
字符串常量池是JVM为了提升性能和减少内存消耗针对字符串专门开辟的一块区域,主要是为了避免重复创建字符串。
JDK1.7将字符串常量池移动到堆中,因为永久代垃圾回收效率太低,只有在整堆收集的时候才会被执行,而大量字符串通常是需要被及时回收的,因此移动到堆中。
直接内存是一种特殊的内存缓冲区,通过JNI的方式在本地内存中分配。
CSDN
内存分配时需要考虑线程安全问题,通常采用两种方式保证线程安全:
对象访问的方式由虚拟机具体实现而定,目前主流的是使用句柄和直接指针两种。
句柄方式如下:堆中划分一块内存作为句柄池,线程栈帧的局部变量表中存储的reference是对象的句柄地址,句柄中又包含了对象实例数据和对象类型数据各自具体的地址信息。
这种方式的优点在于对象被移动时只改变句柄中的信息,而reference本身不需修改。

直接指针访问方式如下:reference中存储的就是对象地址,这种方法节省了一次指针定位的开销。HotSpot虚拟机采用这种方式。

class文件即字节码,是面向JVM的文件,在一定程度上解决了传统解释型语言执行效率低的问题,同时又保留了解释型语言可移植的特点。
ClassFile {
u4 magic; //Class 文件的标志
u2 minor_version;//Class 的小版本号
u2 major_version;//Class 的大版本号
u2 constant_pool_count;//常量池的数量
cp_info constant_pool[constant_pool_count-1];//常量池
u2 access_flags;//Class 的访问标记
u2 this_class;//当前类
u2 super_class;//父类
u2 interfaces_count;//接口数量
u2 interfaces[interfaces_count];//一个类可以实现多个接口
u2 fields_count;//字段数量
field_info fields[fields_count];//一个类可以有多个字段
u2 methods_count;//方法数量
method_info methods[methods_count];//一个类可以有个多个方法
u2 attributes_count;//此类的属性表中的属性数
attribute_info attributes[attributes_count];//属性表集合
}


各组件说明如下:
constant_pool_count-1,计数器从1开始,索引值为0代表不引用任何常量池项。常量池中每一项都代表一个常量。共有:加载、验证、准备、解析、初始化、使用、卸载7个阶段,其中,验证、准备、解析统称为链接阶段。

加载时需要完成以下三件事
确保Class文件的字节流中包含的信息符合约束要求,不会危害虚拟机安全,主要包括:文件格式验证、元数据验证、字节码验证、符号引用验证。

准备阶段是正式为静态变量分配内存并设置初始值的阶段,都在方法区中分配。

解析阶段是虚拟机将常量池内的符号引用替换为直接引用的过程。主要针对类或接口、字段、类方法、接口方法、方法类型、方法句柄和调用限定符7类符号引用进行。
每个类都有一张方法表存放所有的方法地址,当需要调用时只需根据方法表即可直接调用,也就是将符号引用替换为直接引用。
初始化阶段是执行方法的过程,这一步开始才真正开始执行程序的字节码。
卸载类即该类的Class对象被GC。需要满足三个要求才可卸载:
在JVM生命周期内,JVM自带的类加载器加载的类不会被卸载,可以自定义类加载器并进行类卸载。