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

本文涉及的产品
容器镜像服务 ACR,镜像仓库100个 不限时长
简介: 堆对象的生命周期是咋样的?什么时候被回收,回收前又如何流转?具体又是被如何回收?今天重点讲对象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算法我们在下一篇文章分析。

相关文章
|
2月前
|
存储 算法 Java
散列表的数据结构以及对象在JVM堆中的存储过程
本文介绍了散列表的基本概念及其在JVM中的应用,详细讲解了散列表的结构、对象存储过程、Hashtable的扩容机制及与HashMap的区别。通过实例和图解,帮助读者理解散列表的工作原理和优化策略。
43 1
散列表的数据结构以及对象在JVM堆中的存储过程
|
2月前
|
缓存 Prometheus 监控
Elasticsearch集群JVM调优设置合适的堆内存大小
Elasticsearch集群JVM调优设置合适的堆内存大小
355 1
|
28天前
|
存储 监控 算法
深入探索Java虚拟机(JVM)的内存管理机制
本文旨在为读者提供对Java虚拟机(JVM)内存管理机制的深入理解。通过详细解析JVM的内存结构、垃圾回收算法以及性能优化策略,本文不仅揭示了Java程序高效运行背后的原理,还为开发者提供了优化应用程序性能的实用技巧。不同于常规摘要仅概述文章大意,本文摘要将简要介绍JVM内存管理的关键点,为读者提供一个清晰的学习路线图。
|
30天前
|
缓存 监控 算法
Python内存管理:掌握对象的生命周期与垃圾回收机制####
本文深入探讨了Python中的内存管理机制,特别是对象的生命周期和垃圾回收过程。通过理解引用计数、标记-清除及分代收集等核心概念,帮助开发者优化程序性能,避免内存泄漏。 ####
41 3
|
11天前
|
缓存 Java
JVM对象引用
本次课程聚焦JVM对象引用,涵盖强引用、软引用、弱引用和虚引用。强引用是最常见的引用类型,确保对象不会被垃圾回收器回收,适用于需要确保对象存活的场景;软引用在内存不足时会被优先回收,常用于缓存;弱引用的对象随时可能被回收,适合临时对象;虚引用最弱,主要用于接收对象回收通知,进行资源清理。通过合理选择引用类型,可优化内存管理,避免内存泄露。
|
2月前
|
Java
JVM内存参数
-Xmx[]:堆空间最大内存 -Xms[]:堆空间最小内存,一般设置成跟堆空间最大内存一样的 -Xmn[]:新生代的最大内存 -xx[use 垃圾回收器名称]:指定垃圾回收器 -xss:设置单个线程栈大小 一般设堆空间为最大可用物理地址的百分之80
|
2月前
|
Java
JVM运行时数据区(内存结构)
1)虚拟机栈:每次调用方法都会在虚拟机栈中产生一个栈帧,每个栈帧中都有方法的参数、局部变量、方法出口等信息,方法执行完毕后释放栈帧 (2)本地方法栈:为native修饰的本地方法提供的空间,在HotSpot中与虚拟机合二为一 (3)程序计数器:保存指令执行的地址,方便线程切回后能继续执行代码
25 3
|
2月前
|
存储 缓存 监控
Elasticsearch集群JVM调优堆外内存
Elasticsearch集群JVM调优堆外内存
54 1
|
2月前
|
Arthas 监控 Java
JVM进阶调优系列(9)大厂面试官:内存溢出几种?能否现场演示一下?| 面试就那点事
本文介绍了JVM内存溢出(OOM)的四种类型:堆内存、栈内存、元数据区和直接内存溢出。每种类型通过示例代码演示了如何触发OOM,并分析了其原因。文章还提供了如何使用JVM命令工具(如jmap、jhat、GCeasy、Arthas等)分析和定位内存溢出问题的方法。最后,强调了合理设置JVM参数和及时回收内存的重要性。
|
2月前
|
存储 编译器 Linux
【c++】类和对象(上)(类的定义格式、访问限定符、类域、类的实例化、对象的内存大小、this指针)
本文介绍了C++中的类和对象,包括类的概念、定义格式、访问限定符、类域、对象的创建及内存大小、以及this指针。通过示例代码详细解释了类的定义、成员函数和成员变量的作用,以及如何使用访问限定符控制成员的访问权限。此外,还讨论了对象的内存分配规则和this指针的使用场景,帮助读者深入理解面向对象编程的核心概念。
143 4