1.Java运行时数据区
- 方法区,堆线程共享。虚拟机栈,本地方法栈和程序计数器线程私有。
2.程序计数器(PC计数器)
- 占用较小的一块内存空间,当执行Java方法时记录正在执行的虚拟机字节码指令地址,如果执行Native方法则计时器值为空。
3.Java虚拟机栈
- java方法执行时的内存模型
- 查看Java虚拟机栈详细
3.1 栈帧
- 每个方法都会在虚拟机栈中创建一个对应的栈帧,用于存储局部变量表,操作数栈,动态链接,方法出口等信息。
- 一个方法的调用到结束就对应这一个栈帧从虚拟机栈入栈到出栈。
4.栈内存和堆内存
- 一般我们说的栈内存和堆内存的栈就指虚拟机栈内存或者就指局部变量表,堆就是堆内存。最关心的也就是这两块。
5.本地方法栈
- 本地方法栈和虚拟机栈作用相似,虚拟机栈为虚拟机执行java方法(字节码)服务,本地方法栈为jvm使用Native方法服务。
- 本地方法 关键字 native ,这些方法一般用来调用本地方法库中的方法这些方法操作底层大多用C 实现。
- 有的虚拟机将本地方法栈和虚拟机栈合在一起,如HotSpot。
6.java堆
- 最大的一块内存,存放对象实例的地方。
- Java堆是垃圾收集器管理的主要区域,也称GC堆。
- Java堆物理上可不连续,逻辑上连续。
- 堆中没有完成实例分配,并且对也无法在扩展时抛出 OutOfMemoryError异常
7.方法区
- 方法区是一种规范,1.8之前永久代是其一周种实现,1.8开始使用元空间代替永久代。
- 存已经被虚拟机加载的类信息(Class对象)、常量、静态变量、即时编译器JIT编译过后的代码数据。
- jdk 1.7、1.8 对方法区做出了修改,1.8 取消了永久代使用元空间代替。
- 方法区被Java虚拟机规范描述为堆的一个逻辑部分,但它不是堆,有一个别名叫非堆Non-Heap
- jdk1.6 及之前方法区位于永久代(PermGen),永久代和堆相互隔离。
- 永久代使用的是jvm内存,而元空间使用本地内存,随意元空间可以存放更多信息。
- 方法区可以不需要连续的内存空间,也可以固定大小,也可以扩展,也可以不实现垃圾收集,如果实现则主要针对常量池的回收和类型的卸载。
7.4 运行时常量池
- 在方法区,用来存放编译期生成的各种符号引用和字面量。
- 编译期将各种符号引用和字面量放置在class文件的常量池中,解析后在运行时常量池,字符串在方法区的字符串常量池中(1.7之前),方法运行时复制到局部变量表中。1.7开始字符串常量池被移入堆中。
- 符号引用: String ss = "asdsfg" ss这个符号就是符号引用。解析阶段解析为直接引用
- 字面量: 值本身 如 asdfg
- 常量池中有各种虚拟机正常运行需要的字符串,所以有些即使不创建也还会存在于字符串常量池中如"java"。
8.直接内存
- 并非虚拟机运行时的内存,也不再jvm规范中定义。
- 在JDK 1.4中新加入了NIO(New Input/Output)类,引入了一种基于通道(Channel)与缓冲区(Buffer)的I/O方式,它可以使用Native函数库直接分配堆外内存,然后通过一个存储在Java堆中的DirectByteBuffer对象作为这块内存的引用进行操作。这样能在一些场景中显著提高性能,因为避免了在Java堆和Native堆中来回复制数据。
9.对象的创建
- 普通Java对象创建,不包括数组和Class对象。
- 类加载 连接 类加载检查
- 检查通过后为对象分配内存,对象内存大小在类加载时就已经确定了。
- 将分配的内存初始化为0值,这一步保证了实例字段即使不赋初值也可使用。
- 对对象进行设置,如对象哈希码,是哪个类的实例,GC分代年龄。
- <init> 方法执行,对象执行初始化。
10.对象内存分布
- 对象在内存中分为3块区域,对象头(Header)、实例数据(Instance Data)、对齐填充(Padding)。
- 实例数据:包括从父类继承下来的字段,并且将宽度相同的字段分配到一起。
- 对齐填充:HotSpot VM 的内存自动管理系统要求对象起始地址必须是8字节的整数倍,就是说对象大小必须是8字节整数倍,而对象头正好是8字节的倍数,所以实例数据不够8字节整数倍时会补齐填充。
10.1 对象头
- 对象头分为2部分,第一部分存储对象自身运行时数据,第二部分存储类型指针。
- 对象头作用 查看博客
- 自身运行时数据(Mark Word):如哈希码、GC分代年龄、锁转态标志、线程持有的锁,偏向线程ID、偏向时间戳。Mark Word是非固定的数据结构,以便存储更多信息,根据对象状态不同各个信息所占位数会变化,但总体肯定是8字节倍数。
- 类型指针:指向元数据(Class类数据)的指针
- 如果对象是数组那么对象头还有一块用于记录数组长度的区域,因为普通Java对象通过元数据可以确定大小,而数组的元数据无法确定数组大小。
11.对象定位
- 主流的访问对象的方式有两种,通过句柄访问和直接访问,HotSpot 采用直接访问。
- 通过句柄访问就是在堆中有一个句柄池存放对象实例数据与类型数据各自的具体地址,栈中的引用保存句柄的地址,好处是引用保存的地址不会变比较稳定,对象改变也只改变句柄池中的地址。
- 直接访问就是引用直接指向对象,好处就是速度快。