浅谈GC

        垃圾收集器的目的就是让应用程序开发者从人为进行动态内存管理中解放出来。开发者不需要分配和回收内存,不需要密切关注分配的动态内存的生命周期。Java HotSpot VM提供了一系列垃圾收集算法可供选择。

        什么时候选择一个垃圾收集器很重要?

        对于一些应用来说无所谓,也就是说在垃圾收集频率和暂停时间适中的情况下应用也可以运行的很好。然而,对于大型应用来说情况就不是这样的了,尤其是那些大数据量、多线程和很高TPS的系统。阿姆达尔定律说,并不是所用的任务都能很好的并行化,某些部分总是串行的。

在Java平台中,目前有四个支持的垃圾收集替代方案,除了其中一个serial GC之外,其他的都是并行工作,以提高性能。尽可能降低垃圾回收的开销是非常重要的

对于一些小的应用来说,串行收集器是很好的,而当应用部署在服务器类型的机器上时,默认情况下会选择Garbage-First (G1) collector。

On server-class machines, the following are selected by default:
• Garbage-First (G1) collector
• Initial heap size of 1/64 of physical memory
• Maximum heap size of 1/4 of physical memory

• Tiered compiler, using both C1 and C2

**基于行为的调优:Java HotSpot VM垃圾收集器可以配置为优先满足两个目标其中之一:最大暂停时间应用程序吞吐量

1.Maximum Pause-Time Goal

暂停时间是垃圾收集器停止应用程序的持续时间并回收不再使用的空间。这个目标的用意就是尽量缩短暂停时间。

可以通过命令行选项 -XX:MaxGCPauseMillis=<nnn>指定这个目标,指定了之后,垃圾收集器会调整堆大小和其他参数来尽量让暂停时间达到这个目标值。不过这些调整可能会导致GC的频率增加,从而会减少应用的整体吞吐量。

2.Throughput Goal

        吞吐量目标是根据收集垃圾所花费的时间来衡量的,垃圾收集以外的时间就是应用程序的时间。这个目标是通过命令行选项-XX:GCTimeRatio=nnn指定的。垃圾收集时间与应用程序时间的比率为 1/ (1+nnn)。例如,-XX:GCTimeRatio=19表明1/20或者总时间的5%用于垃圾收集。花费在垃圾收集上的时间是所有垃圾收集引起暂停的总时间。如果吞吐量目标未被满足,那么垃圾收集器可能会增加堆的大小,这样在两次垃圾收集暂停之间的应用程序时间会更长,吞吐量相应的会提高。


**调优策略

1.不要为堆选择最大值除非你知道你需要一个大于默认最大堆大小的堆。为您的应用选择足够的吞吐量目标。

2.应用程序行为的更改可能会导致堆增大或缩小。例如,如果应用程序开始以更高的速率分配内存,则堆会增长保持相同的吞吐量。

3.如果堆增长到最大尺寸并且吞吐量目标未达到,那么最大堆空间对于设定的吞吐量目标来说太小。将最大堆大小设置为接近平台上总物理内存的值,但不会导致应用程序交换。如果吞吐量目标仍然没有达到,那么想要的应用程序时间目标对于平台上的可用内存来说太高。

4.如果吞吐量目标可以达到,但暂停时间太长,则设定一个最大暂停时间的目标。选择最大暂停时间目标可能意味着你的吞吐量目标将不会被满足,所以为应用程序选择可接受的折中值。

5.一般情况下,在垃圾收集器尽力满足这两个竞争目标时,堆的大小是不断波动的。在应用程序达到一个稳定状态时也会有这种情况。想达到吞吐量目标需要一个大堆,想达到maximum pause-time和minimum footprint(最小占用空间)都需要一个小堆。


<think>嗯,用户想了解JVM的工作原理和基本概念。首先,我需要回顾一下JVM的基本结构和运行机制。根据用户提供的引用,JVM主要包括类加载子系统、运行时数据区、执行引擎和本地方法接口等部分。运行时数据区又分为方法区、堆、虚拟机栈、本地方法栈和程序计数器。用户之前提到的引用中提到了堆、栈、方法区等,所以需要详细说明这些部分。 首先,类加载过程可能需要分步骤解释,加载、链接、初始化这三个阶段。然后运行时数据区中的各个部分,比如堆是对象实例存储的地方,是GC的主要区域,分代结构(年轻代、老年代、永久代/元空间)。虚拟机栈和本地方法栈的区别,栈帧的结构,局部变量表、操作数栈这些内容。程序计数器的作用,线程私有,记录当前执行地址。 执行引擎部分需要解释解释器和JIT编译器,以及它们的协作。本地方法接口用本地库。垃圾回收机制的基本原理,分代收集算法,不同区域的GC策略,比如年轻代的Minor GC,老年代的Major GC,还有Full GC。 可能还需要提到JVM的内存管理,比如如何分配内存,对象的创建和回收过程。此外,用户提供的引用中有提到永久存储区(方法区),在JDK8之后被元空间替代,使用本地内存,避免OOM。 用户的问题需要结构清晰的回答,所以应该分点讲解,每个部分简明扼要。同时,生成的相关问题要围绕JVM的结构、内存管理、类加载、垃圾回收等方面,确保覆盖用户可能感兴趣的后续问题。 在回答时要注意使用正确的中文术语,比如“栈帧”而不是“栈框架”,确保LaTeX格式正确,行内公式用$...$,独立公式用$$...$$。比如在讲解对象内存分配时,可能需要公式表示内存地址,但用户提供的引用中没有具体公式,所以可能不需要太多数学表达式,但如果有涉及的话要正确应用。 最后,检查引用是否正确标注,例如方法区的信息引用[4],堆的结构引用[3],栈的局部变量引用[2]等。确保每个引用标记自然,不干扰阅读。同时,生成的问题要有助于用户深入理解,比如JVM内存结构、类加载机制、垃圾回收化、元空间与永久代的区别等。</think>### JVM工作原理与基本概念详解 --- #### **一、JVM核心组成** Java虚拟机(JVM)是Java程序运行的核心环境,主要包含以下模块: 1. **类加载子系统** 负责加载`.class`文件,解析为JVM可识别的数据结构,过程分为三个阶段: - **加载**:查找并加载类的二进制数据。 - **链接**:验证、准备(分配静态变量内存)、解析(符号引用转直接引用)。 - **初始化**:执行类构造器`<clinit>()`方法[^4]。 2. **运行时数据区** JVM内存分为以下区域: - **方法区(Method Area)** 存储类元数据(如类名、方法代码)、静态变量、常量池等。JDK8后由“永久代”改为“**元空间(Metaspace)**”,直接使用本地内存[^4]。 - **堆(Heap)** 所有线程共享,存储对象实例和数组,是垃圾回收的主战场。采用分代结构: - 年轻代(Young Generation):存放新对象,分为Eden、Survivor区。 - 老年代(Old Generation):长期存活的对象。 $$ \text{堆内存} = \text{年轻代} + \text{老年代} $$ [^3] - **虚拟机栈(JVM Stack)** 线程私有,存储栈帧(Frame),每个栈帧对应一个方法用,包含局部变量表、操作数栈等[^2]。 - **本地方法栈(Native Method Stack)** 为本地方法(如C/C++代码)服务。 - **程序计数器(PC Register)** 记录当前线程执行指令的位置。 3. **执行引擎** - **解释器**:逐行解释字节码。 - **即时编译器(JIT)**:将热点代码编译为机器码以提升性能。 - **垃圾回收器(GC)**:自动回收堆中无用对象。 4. **本地方法接口(JNI)** 用操作系统或硬件相关的本地库。 --- #### **二、JVM运行流程** 1. **类加载**:将`.class`文件加载至方法区。 2. **内存分配**:对象实例分配在堆中,局部变量存储于虚拟机栈。 - 例如:`Object obj = new Object();`中,`obj`引用在栈中,对象实例在堆中[^2][^3]。 3. **执行字节码**:解释器或JIT将字节码转换为机器指令。 4. **垃圾回收**:GC自动回收堆中不可达对象,避免内存泄漏。 --- #### **三、垃圾回收(GC)原理** - **分代收集算法**: - **年轻代**:使用复制算法(Minor GC),Eden区满时触发。 - **老年代**:使用标记-清除或标记-整理算法(Major GC/Full GC)[^3]。 - **GC Roots**:通过根对象(如栈中局部变量、静态变量)判断对象存活性。 --- #### **四、关键特性** 1. **跨平台性**:字节码通过JVM适配不同操作系统。 2. **内存自动管理**:GC减少内存泄漏风险。 3. **多线程支持**:独立栈和程序计数器实现线程隔离。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值