本篇涵盖一下内容
- jvm内存区域的用途介绍
- jvm堆的区域结构
jvm内存区域的用途介绍
java虚拟机在执行java程序时会把它管理的内存划分为不同的区域,每个区域有不同的用途。
该图片来源于《实战Java虚拟机:JVM故障诊断与性能优化(第二版)》。以下是各区域的简单介绍
- 类加载子系统负责从文件系统或者网络中加载Class信息,加载的类信息存放于一块称为方法区的内存空间中。除了类的信息,方法区中可能还会存放运行时常量池信息,包括字符串字面量和数字常量(这部分常量信息是Class文件中常量池部分的内存映射)。方法区就是Metaspace。
- Java堆在虚拟机启动的时候建立,它是Java程序最主要的内存工作区域。几乎所有的Java对象实例都存放于Java堆中。堆空间是所有线程共享的,这是一块与Java应用密切相关的内存区域。
- Java的NIO库允许Java程序使用直接内存。直接内存是在Java堆外的、直接向系统申请的内存区域。通常,访问直接内存的速度会优于Java堆。因此,出于性能考虑,读写频繁的场合可能会考虑使用直接内存。由于直接内存在Java堆外,因此,它的大小不会直接受限于Xmx指定的最大堆大小,但是系统内存是有限的,Java堆和直接内存的总和依然受限于操作系统的最大内存。
- 垃圾回收系统是Java虚拟机的重要组成部分,垃圾回收器可以对方法区、Java堆和直接内存进行回收。其中,Java堆是垃圾收集器的工作重点。和C/C++不同,Java中所有的对象空间释放都是隐式的。也就是说,Java中没有类似free()或者delete()这样的函数释放指定的内存区域。对于不再使用的垃圾对象,垃圾回收系统会在后台默默工作,默默查找、标识并释放垃圾对象,完成包括Java堆、方法区和直接内存中的全自动化管理。
- 每一个Java虚拟机线程都有一个私有的Java栈。一个线程的Java栈在线程创建的时候被创建。Java栈中保存着帧信息,Java栈中保存着局部变量、方法参数,同时和Java方法的调用、返回密切相关。
- 本地方法栈和Java栈非常类似,最大的不同在于Java栈用于Java方法的调用,而本地方法栈则用于本地方法的调用。作为对Java虚拟机的重要扩展,Java虚拟机允许Java直接调用本地方法(通常使用C语言编写)。
- PC(ProgramCounter)寄存器也是每个线程私有的空间,Java虚拟机会为每一个Java线程创建PC寄存器。在任意时刻,一个Java线程总是在执行一个方法,这个正在被执行的方法称为当前方法。如果当前方法不是本地方法,PC寄存器就会指向当前正在被执行的指令。如果当前方法是本地方法,那么PC寄存器的值就是undefined。
- 执行引擎是Java虚拟机的最核心组件之一,它负责执行虚拟机的字节码。现代虚拟机为了提高执行效率,会使用即时编译技术将方法编译成机器码后再执行。
jvm堆的区域结构
jvm内存区域中,用来存储对象的区域叫做jvm堆(JVM HEAP),该区域与对象的创建和回收息息相关,我们先了解一下jvm堆内部的结构,下面会介绍常用的GC算法。
从整体上上看,jvm堆可以分为两大部分:
- 新生代(Young)
- 老年代(Old/ Tenured)
根据垃圾回收机制的不同,Java堆有可能拥有不同的结构。最为常见的一种构成是将整个Java堆分为新生代和老年代。其中,新生代存放新生对象或者年龄不大的对象,老年代则存放老年对象。新生代一般分为eden、s0、s1,其中s0和s1也被称为from和to区域,它们是两块大小相等、可以互换角色的内存空间。
在绝大多数情况下,对象首先在eden区分配,在一次新生代回收后,如果对象还存活,则会进入s0或者s1,之后,每经过一次新生代回收,对象如果存活,它的年龄就会加1。当对象的年龄达到一定条件后,就会被认为是老年对象,从而进入老年代。