前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >JVM进阶调优系列(3)堆内存的对象什么时候被回收?

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

原创
作者头像
拉丁解牛说技术
发布2024-10-17 17:58:44
960
发布2024-10-17 17:58:44
举报
文章被收录于专栏:JVM高手修炼

上文末尾留了一道题给大家思考:堆对象的生命周期是咋样的?什么时候被回收,回收前又如何流转?具体又是被如何回收?今天重点讲对象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,哪些对象会被回收哪些不能回收?

代码语言:txt
复制
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算法我们在下一篇文章分析。

原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 一、虚拟机栈会发生GC吗?
  • 二、什么时候会发生YGC?
  • 三、以下代码发生了YGC,哪些对象会被回收哪些不能回收?
  • 四、年轻代GC存活的对象什么时候进入老年代?
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档