JVM内存区域划分与核心职责
Java虚拟机在执行Java程序的过程中会将其所管理的内存划分为若干个不同的数据区域。这些区域各有其独特的职责,共同协作以支持程序的运行。主要包括程序计数器、Java虚拟机栈、本地方法栈、堆以及方法区。程序计数器是一块较小的内存空间,它可以看作是当前线程所执行的字节码的行号指示器,是线程私有的。Java虚拟机栈描述的是Java方法执行的内存模型,每个方法在执行的同时都会创建一个栈帧用于存储局部变量表、操作数栈、动态链接、方法出口等信息,它同样是线程私有的。本地方法栈则为虚拟机使用到的Native方法服务。
堆内存的详细解析
堆是Java虚拟机所管理的内存中最大的一块,被所有线程共享,在虚拟机启动时创建。此内存区域的唯一目的就是存放对象实例,几乎所有的对象实例以及数组都在这里分配内存。Java堆是垃圾收集器管理的主要区域,因此很多时候也被称作“GC堆”。从内存回收的角度来看,由于现代收集器基本都采用分代收集算法,所以Java堆中还可以细分为:新生代和老年代;再细致一点的有Eden空间、From Survivor空间、To Survivor空间等。从内存分配的角度来看,线程共享的Java堆中可能划分出多个线程私有的分配缓冲区(Thread Local Allocation Buffer, TLAB)。
垃圾收集的核心算法与策略
垃圾收集(Garbage Collection, GC)是JVM内存管理的核心,主要负责自动回收不再被使用的对象所占用的内存。其核心问题在于如何判断对象是否存活,主要算法有引用计数法和可达性分析算法。主流的商用程序语言(如Java、C#)的主流实现中,都是通过可达性分析(Reachability Analysis)来判定对象是否存活的。垃圾收集算法则主要包括标记-清除算法、复制算法、标记-整理算法以及分代收集算法。现代商用虚拟机大多采用分代收集理论,针对堆内存中不同生命周期的对象采取不同的收集策略,例如在新生代使用复制算法,而在老年代则使用标记-清除或标记-整理算法。
内存分配与回收机制实践
对象的内存分配,从概念上讲,基本上都是在堆上进行的,但细致分析后,也可能在栈上分配(标量替换)。对象优先在新生代的Eden区分配,当Eden区没有足够空间进行分配时,虚拟机将发起一次Minor GC。大对象(需要大量连续内存空间的Java对象,如很长的字符串以及数组)会直接进入老年代,以避免在Eden区及两个Survivor区之间发生大量的内存复制。长期存活的对象(经过多次Minor GC后仍然存活)也将逐渐从新生代晋升到老年代。除了对象年龄之外,为了适应不同程序的内存状况,虚拟机还提供了动态对象年龄判定和空间分配担保等机制。
方法区与垃圾收集
方法区与Java堆一样,是各个线程共享的内存区域,它用于存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据。Java虚拟机规范对方法区的限制非常宽松,除了和Java堆一样不需要连续的内存和可以选择固定大小或者可扩展外,还可以选择不实现垃圾收集。这区域的内存回收目标主要是针对常量池的回收和对类型的卸载,但通常回收条件较为苛刻,回收效果也远不及Java堆。在JDK 8及之后,永久代(PermGen)被元空间(Metaspace)所取代,元空间使用本地内存,其最大可分配空间就是系统可用内存空间,从而减少了OOM异常的发生。
常见的垃圾收集器
HotSpot虚拟机提供了多种不同的垃圾收集器,每种都有其特点和适用场景。Serial收集器是一个单线程的收集器,它在进行垃圾收集时,必须暂停所有其他工作线程(Stop The World),适用于客户端模式下的虚拟机。Parallel Scavenge收集器是一个新生代收集器,它也是使用复制算法的并行收集器,它的关注点是达到一个可控制的吞吐量。CMS(Concurrent Mark Sweep)收集器是一种以获取最短回收停顿时间为目标的收集器,它基于标记-清除算法实现,允许垃圾收集线程与用户线程并发工作。G1(Garbage-First)收集器是面向服务端应用的垃圾收集器,它将堆内存划分为多个大小相等的独立区域(Region),并优先回收垃圾最多的区域,同时兼顾了低停顿和高吞吐量的目标。
286

被折叠的 条评论
为什么被折叠?



