JVM进阶调优系列(3)堆内存的对象什么时候被回收?

本文涉及的产品
容器镜像服务 ACR,镜像仓库100个 不限时长
容器服务 Serverless 版 ACK Serverless,952元额度 多规格
容器服务 Serverless 版 ACK Serverless,317元额度 多规格
简介: 堆对象的生命周期是咋样的?什么时候被回收,回收前又如何流转?具体又是被如何回收?今天重点讲对象GC,看完这篇就全都明白了。

上文末尾留了一道题给大家思考:堆对象的生命周期是咋样的?什么时候被回收,回收前又如何流转?具体又是被如何回收?今天重点讲对象GC,看完这篇就清晰了。

一、虚拟机栈会发生GC吗?

     上一篇文章说到虚拟机栈,虚拟机栈里有局部变量,这个局部变量引用了一个对象,这个对象在堆内存是要被GC回收的,但是虚拟机栈要不要做GC呢,不GC那这个局部变量怎么办?

     实际是不用GC的。线程的虚拟机栈,就是块单独的内存,和堆内存不在一起。虚拟机栈的局部变量,在出栈的时候,就从栈内存里移除去了,出栈完,这个栈内存就清空。所以不需要GC。


二、什么时候会发生YGC?

当JVM想把在年轻代里新增一个对象,但是年轻代空间不足,就会发生YGC。那年轻代空间为何不足呢,如何设置年轻代的大小。我们看一下JVM内存的核心参数。

-Xms:堆内存,最开始堆内存初始大小。

-Xmx:堆内存可以扩张到最大的大小。

-Xmn:堆内存里,年轻代占用的大小,堆内存大小-年轻代大小=就是老年代的空间大小。年轻代里上一篇文章说过里面分:Eden区、SurvivorTo区和SurvivorFrom区,他们按8:1:1的比例划分年轻代的空间。

-XX:MetaspaceSize:元数据空间大小,在JDK8之前对应的是永久代PermSize。

-XX:MaxPermSize:元数据空间可以扩张的最大大小,在JDK8之前对应的永久代MaxPermSize。

-Xss:每个线程虚拟机栈的大小。


所以当-Xmn年轻代里的Eden区空闲内存无法存放新对象,也就是空间不够,这时候就触发YGC,需要将部分对象回收,无法回收的挪到老年代腾出空间。


三、以下代码发生了YGC,哪些对象会被回收哪些不能回收?

package lading.java.jvm;
import java.lang.ref.PhantomReference;
import java.lang.ref.ReferenceQueue;
import java.lang.ref.SoftReference;
import java.lang.ref.WeakReference;
/**
 * 说说这个.class 加载到jvm经历的过程,以及在执行过程中,内存如何分配,GC的详细过程
 */
public class Demo002JvmShow {
    public static final String name = "我是类静态变量";
    public static User userA = new User("A");
    public static SoftReference<User> userB = new SoftReference<>(new User("B"));//软引用
    public static WeakReference<User> userC = new WeakReference<>(new User("C"));//弱引用
    public static PhantomReference<Object> objectD = new PhantomReference<>(new Object(),new ReferenceQueue<>());//虚引用
    private int paramInt = 1000;//八大基本类型
    private Object object = new Object();
    private boolean isOk = 10 / 2 == 4 ? true : false;
    public static void main(String[] args) {
        //循环,让堆内存发生YGC
        while (true) {
            Demo002JvmShow show = new Demo002JvmShow();
            User user = new User("拉丁");
            show.paramInt = user.getAge();
        }
    }
}

      JVM 采用的是【可达性分析算法】来判断一个对象是否可以被回收。这个算法的核心就是,通过判断对象是否还有被引用(这里又分强引用、软引用弱引用、虚引用而不同),也就是判断对象是否有GC Roots来进行回收。

      如果你的对象被局部变量、或者类的静态变量引用,GC就不会回收。比如当前执行到方法的show.paramInt = user.getAge(); 里面对象user,由于有main线程的虚拟机栈的user变量引用,这时候YGC是不会回收堆内存里的User对象。



然后对象里有很多变量,来看看哪些会被回收?


强引用的比如name,不能回收。

软引用的userB 对象可能会被回收,因为软引用,如果GC后空间够用就软引用的对象不会被回收,但是GC后空间还不够,那不好意思,软引用的对象也要被回收。

弱引用的userC,被回收。弱引用的对象,一旦发生GC,就会被回收。

如果一个对象没有GC Roots,是否也会被立马回收?

这个不一定。比如对象重写了finalize方法,并赋予新值。GC回收该对象之前,会调用它的finalize方法,如果它还有其他引用尤其是引用了自己,那就不能回收。

四、年轻代GC存活的对象什么时候进入老年代?

     年轻代的对象或者说新对象进入老年代主要有以下3个机制来控制。

    1、动态对象年龄判断机制。

           默认经过15次YGC ,也就是对象的年龄大于15之后,就被挪到老年代。

     2、空间担保机制。

           如果YGC后发现存活的对象比较多,且无法让新对象放到年轻代,然后老年代那边又有足够空间,能保证存下存活的全部对象,就把这些存活对象晋升到老年代存放。

     3、大对象

          如果新增的对象特别大,那这个大对象直接进入老年代。


今天分享到这,具体GC算法我们在下一篇文章分析。

相关文章
|
12天前
|
存储 算法 Java
散列表的数据结构以及对象在JVM堆中的存储过程
本文介绍了散列表的基本概念及其在JVM中的应用,详细讲解了散列表的结构、对象存储过程、Hashtable的扩容机制及与HashMap的区别。通过实例和图解,帮助读者理解散列表的工作原理和优化策略。
26 1
散列表的数据结构以及对象在JVM堆中的存储过程
|
6天前
|
Arthas 监控 Java
JVM进阶调优系列(9)大厂面试官:内存溢出几种?能否现场演示一下?| 面试就那点事
本文介绍了JVM内存溢出(OOM)的四种类型:堆内存、栈内存、元数据区和直接内存溢出。每种类型通过示例代码演示了如何触发OOM,并分析了其原因。文章还提供了如何使用JVM命令工具(如jmap、jhat、GCeasy、Arthas等)分析和定位内存溢出问题的方法。最后,强调了合理设置JVM参数和及时回收内存的重要性。
|
5天前
|
监控 Java 编译器
Java虚拟机调优实战指南####
本文深入探讨了Java虚拟机(JVM)的调优策略,旨在帮助开发者和系统管理员通过具体、实用的技巧提升Java应用的性能与稳定性。不同于传统摘要的概括性描述,本文摘要将直接列出五大核心调优要点,为读者提供快速预览: 1. **初始堆内存设置**:合理配置-Xms和-Xmx参数,避免频繁的内存分配与回收。 2. **垃圾收集器选择**:根据应用特性选择合适的GC策略,如G1 GC、ZGC等。 3. **线程优化**:调整线程栈大小及并发线程数,平衡资源利用率与响应速度。 4. **JIT编译器优化**:利用-XX:CompileThreshold等参数优化即时编译性能。 5. **监控与诊断工
|
16天前
|
存储 监控 Java
JVM进阶调优系列(8)如何手把手,逐行教她看懂GC日志?| IT男的专属浪漫
本文介绍了如何通过JVM参数打印GC日志,并通过示例代码展示了频繁YGC和FGC的场景。文章首先讲解了常见的GC日志参数,如`-XX:+PrintGCDetails`、`-XX:+PrintGCDateStamps`等,然后通过具体的JVM参数和代码示例,模拟了不同内存分配情况下的GC行为。最后,详细解析了GC日志的内容,帮助读者理解GC的执行过程和GC处理机制。
|
23天前
|
Arthas 监控 数据可视化
JVM进阶调优系列(7)JVM调优监控必备命令、工具集合|实用干货
本文介绍了JVM调优监控命令及其应用,包括JDK自带工具如jps、jinfo、jstat、jstack、jmap、jhat等,以及第三方工具如Arthas、GCeasy、MAT、GCViewer等。通过这些工具,可以有效监控和优化JVM性能,解决内存泄漏、线程死锁等问题,提高系统稳定性。文章还提供了详细的命令示例和应用场景,帮助读者更好地理解和使用这些工具。
|
29天前
|
监控 架构师 Java
JVM进阶调优系列(6)一文详解JVM参数与大厂实战调优模板推荐
本文详述了JVM参数的分类及使用方法,包括标准参数、非标准参数和不稳定参数的定义及其应用场景。特别介绍了JVM调优中的关键参数,如堆内存、垃圾回收器和GC日志等配置,并提供了大厂生产环境中常用的调优模板,帮助开发者优化Java应用程序的性能。
|
30天前
|
Java
JVM进阶调优系列(5)CMS回收器通俗演义一文讲透FullGC
本文介绍了JVM中CMS垃圾回收器对Full GC的优化,包括Stop the world的影响、Full GC触发条件、GC过程的四个阶段(初始标记、并发标记、重新标记、并发清理)及并发清理期间的Concurrent mode failure处理,并简述了GC roots的概念及其在GC中的作用。
|
1月前
|
存储 安全 Java
jvm 锁的 膨胀过程?锁内存怎么变化的
【10月更文挑战第3天】在Java虚拟机(JVM)中,`synchronized`关键字用于实现同步,确保多个线程在访问共享资源时的一致性和线程安全。JVM对`synchronized`进行了优化,以适应不同的竞争场景,这种优化主要体现在锁的膨胀过程,即从偏向锁到轻量级锁,再到重量级锁的转变。下面我们将详细介绍这一过程以及锁在内存中的变化。
37 4
|
4天前
|
Java Linux Windows
JVM内存
首先JVM内存限制于实际的最大物理内存,假设物理内存无限大的话,JVM内存的最大值跟操作系统有很大的关系。简单的说就32位处理器虽然可控内存空间有4GB,但是具体的操作系统会给一个限制,这个限制一般是2GB-3GB(一般来说Windows系统下为1.5G-2G,Linux系统下为2G-3G),而64bit以上的处理器就不会有限制。
8 1
|
1月前
|
缓存 算法 Java
JVM知识体系学习六:JVM垃圾是什么、GC常用垃圾清除算法、堆内存逻辑分区、栈上分配、对象何时进入老年代、有关老年代新生代的两个问题、常见的垃圾回收器、CMS
这篇文章详细介绍了Java虚拟机(JVM)中的垃圾回收机制,包括垃圾的定义、垃圾回收算法、堆内存的逻辑分区、对象的内存分配和回收过程,以及不同垃圾回收器的工作原理和参数设置。
62 4
JVM知识体系学习六:JVM垃圾是什么、GC常用垃圾清除算法、堆内存逻辑分区、栈上分配、对象何时进入老年代、有关老年代新生代的两个问题、常见的垃圾回收器、CMS