技术学习

《深入理解Java虚拟机》-Java内存区域

2018-10-28  本文已影响2人  Waisam

本文大部分摘自《深入理解Java虚拟机(第二版)》

1. 运行时数据区域

Java虚拟机在执行Java程序的过程中会把它所管理的内存划分为若干个不同的数据区域。Java虚拟机所管理的内存将会包括以下几个运行时数据区域,如下图所示:


Java虚拟机运行时内存模型.png

1.1 程序计数器

程序计数器是一块较小的内存空间,可以看作是当前线程所执行的字节码的行号指示器。字节码解释器通过改变计数器的值来选取下一条需要执行的字节码指令。

1.2 Java虚拟机栈

Java虚拟机栈也是线程私有,生命周期和线程相同。虚拟机栈描述的是Java方法执行的内存模型:每个方法被执行的时候都会创建一个栈帧,用于存储局部变量表、操作数栈、动态连接、方法出口等信息。

1.3 本地方法栈

本地方法栈为虚拟机使用到的Native方法服务,而虚拟机栈为虚拟机执行Java方法服务。

1.4 Java堆

Java堆是被所有线程共享的一块内存区域,在虚拟机启动时创建。几乎所有的对象实例都在堆上分配内存。Java堆可以细分为新生代和老年代,新生代包括Eden区、From Survivor区和To Survivor区。

1.5 方法区

方法区和Java堆一样是各个线程共享的内存区域,用于存储已经被虚拟机加载的类信息、常量、静态常量、即时编译器编译后的代码数据等数据。

Hotspot虚拟机使用永久代来实现方法区,永久代和方法区并不等价。JDK1.7已经将常量池从永久代移出。

1.6 运行时常量池

运行时常量池是方法区的一部分。Class文件中除了有类的版本、字段、方法、接口等描述信息外,还有常量池用于存放编译期生成的各种字面量和符号引用。这部分内容将在类加载后进入方法区的运行时常量池存放。运行时常量池相对于Class文件常量池一个重要的特性就是具备动态性。参考String类intern的使用。

1.7 直接内存

2. Hotspot虚拟机对象

2.1 对象的创建

虚拟机遇到new指令:

  1. 检查指令的参数能否在常量池中定位到一个类的符号引用。
  2. 检查符号引用代表的类是否已被加载、解析和初始化过。

如果没有,先执行类的加载过程。

  1. 类加载检查通过后,为新生对象分配内存。对象所需内存大小在类加载完成后便可完全确定。内存分配方式有以下两种:
  1. 内存分配完成后,虚拟机需要对分配到的内存空间都初始化为零值(不包括对象头)。
  2. 虚拟机对对象进行必要的设置。例如这个对象是哪个类的实例、如何才能找到类的元数据信息、对象的哈希码、对象的GC分代年龄等信息。这些信息存放在对象头。
  3. 执行new指令后会接着执行init方法,进行初始化。

2.2 对象的内存布局

对象的内存布局可以分为3块区域:对象头(Header)、实例数据(Instance Date)和对齐数据(Padding)。

2.2.1 对象头

2.2.2 实例数据

实例数据是对象真正存储的有效信息,也是程序代码中所定义的各种类型的字段内容。包括父类继承和子类定义。存储顺序受虚拟机分配策略参数和源码中定义顺序影响。

2.2.3 对齐数据

对齐补充并不是必然存在,仅起着占位符的作用。对象大小必须是8字节的整数倍。对象头部分正好是8字节的倍数(1倍或2倍)。因此当对象实例数据没有对齐时,需要通过对齐填充来不全。

2.3 对象的访问定位

Java程序需要通过栈上的reference数据来操作堆上的具体对象。目前主流的访问方式有使用句柄和直接指针两种。

2.3.1 使用句柄

Java堆中划分出一块内存作为句柄池,reference对象存储的是句柄地址。句柄中包含对象实例数据和对象类型数据的指针。如下图所示:


使用句柄访问对象.png

2.3.2 直接指针

如果使用直接指针方式访问,Java堆对象的布局就必须考虑如何放置访问类型数据的相关信息。reference直接存储对象的地址。如下图所示:


使用直接指针访问对象.png

2.3.3 总结

上一篇下一篇

猜你喜欢

热点阅读