• 图解 Java 垃圾回收机制,写得非常好!


    640

    Java技术栈

    www.javastack.cn

    优秀的Java技术公众号

    原文:https://www.oracle.com/webfolder/technetwork/tutorials/obe/java/gc01/index.html

    译文:https://www.oschina.net/translate/java-gc

    翻译:Rhys_Lee, AzureSora, 溪边九节, 小小菜鸟鸡

    什么是自动垃圾回收?

    自动垃圾回收是一种在堆内存中找出哪些对象在被使用,还有哪些对象没被使用,并且将后者删掉的机制。

    所谓使用中的对象(已引用对象),指的是程序中有指针指向的对象;而未使用中的对象(未引用对象),则没有被任何指针给指向,因此占用的内存也可以被回收掉。

    在用 C 之类的编程语言时,程序员需要自己手动分配和释放内存。而 Java 不一样,它有垃圾回收器,释放内存由回收器负责。本文接下来将介绍垃圾回收机制的基本过程。

    第一步:标记

    垃圾回收的第一步是标记。垃圾回收器此时会找出哪些内存在使用中,还有哪些不是。640?wx_fmt=png

    上图中,蓝色表示已引用对象,橙色表示未引用对象。垃圾回收器要检查完所有的对象,才能知道哪些有被引用,哪些没。如果系统里所有的对象都要检查,那这一步可能会相当耗时间。关注Java技术栈微信公众号,回复:JVM46,可以获取一份超全 JVM 调优攻略。

    第二步:清除

    这一步会删掉标记出的未引用对象。640?wx_fmt=png

    内存分配器会保留指向可用内存的引用,以供分配新对象。

    压缩

    为了提升性能,删除了未引用对象后,还可以将剩下的已引用对象放在一起(压缩),这样就能更简单快捷地分配新对象了。640?wx_fmt=png

    为什么需要分代垃圾收集?

    之前说过,逐一标记和压缩 Java 虚拟机里的所有对象非常低效:分配的对象越多,垃圾回收需时就越久。不过,根据统计,大部分的对象,其实用没多久就不用了。JVM 与 Linux 的内存关系详解,这篇推荐阅读。

    来看个例子吧。(下图中,竖轴代表已分配的字节,而横轴代表程序运行时间)

    640?wx_fmt=png

    上图可见,存活(没被释放)的对象随运行时间越来越少。而图中左侧的那些峰值,也表明了大部分对象其实都挺短命的。

    JVM 分代

    根据之前的规律,就可以用来提升 JVM 的效率了。方法是,把堆分成几个部分(就是所谓的分代),分别是新生代、老年代,以及永生代。JVM运行时区域详解,这篇推荐大家看下。640?wx_fmt=png

    新对象会被分配在新生代内存。一旦新生代内存满了,就会开始对死掉的对象,进行所谓的小型垃圾回收过程。一片新生代内存里,死掉的越多,回收过程就越快;至于那些还活着的对象,此时就会老化,并最终老到进入老年代内存。

    Stop the World 事件 —— 小型垃圾回收属于一种叫 "Stop the World" 的事件。在这种事件发生时,所有的程序线程都要暂停,直到事件完成(比如这里就是完成了所有回收工作)为止。

    老年代用来保存长时间存活的对象。通常,设置一个阈值,当达到该年龄时,年轻代对象会被移动到老年代。最终老年代也会被回收。这个事件成为 Major GC。

    Major GC 也会触发STW(Stop the World)。通常,Major GC会慢很多,因为它涉及到所有存活对象。所以,对于响应性的应用程序,应该尽量避免Major GC。还要注意,Major GC的STW的时长受年老代垃圾回收器类型的影响。

    永久代包含JVM用于描述应用程序中类和方法的元数据。永久代是由JVM在运行时根据应用程序使用的类来填充的。此外,Java SE类库和方法也存储在这里。

    如果JVM发现某些类不再需要,并且其他类可能需要空间,则这些类可能会被回收。

    世代垃圾收集过程

    现在你已经理解了为什么堆被分成不同的代,现在是时候看看这些空间是如何相互作用的。 后面的图片将介绍JVM中的对象分配和老化过程。关注Java技术栈微信公众号,回复:JVM46,可以获取一份超全 JVM 调优攻略。

    首先,将任何新对象分配给 eden 空间。 两个 survivor 空间都是空的。

    640?wx_fmt=png

    当 eden 空间填满时,会触发轻微的垃圾收集。

    640?wx_fmt=png

    引用的对象被移动到第一个 survivor 空间。 清除 eden 空间时,将删除未引用的对象。

    640?wx_fmt=png

    在下一次Minor GC中,Eden区也会做同样的操作。删除未被引用的对象,并将被引用的对象移动到Survivor区。然而,这里,他们被移动到了第二个Survivor区(S1)。

    此外,第一个Survivor区(S0)中,在上一次Minor GC幸存的对象,会增加年龄,并被移动到S1中。待所有幸存对象都被移动到S1后,S0和Eden区都会被清空。注意,Survivor区中有了不同年龄的对象。


    640?wx_fmt=png

    在下一次Minor GC中,会重复同样的操作。不过,这一次Survivor区会交换。被引用的对象移动到S0,。幸存的对象增加年龄。Eden区和S1被清空。

    640?wx_fmt=png

     此幻灯片演示了 promotion。 在较小的GC之后,当老化的物体达到一定的年龄阈值(在该示例中为8)时,它们从年轻一代晋升到老一代。


    640?wx_fmt=png

    随着较小的GC持续发生,物体将继续被推广到老一代空间。

    640?wx_fmt=png

    所以这几乎涵盖了年轻一代的整个过程。 最终,将主要对老一代进行GC,清理并最终压缩该空间。

    640?wx_fmt=png

    关注Java技术栈微信公众号,在后台回复关键字:Java,可以获取一份栈长整理的 Java 最新技术干货。

    最近干货分享

    Eclipse 最常用的 10 组快捷键,个个牛逼!

    Spring Cloud Eureka 自我保护机制实战分析

    绝对 "牛X" 的代码注释,喜欢就拿去用!

    微服务配置中心对比,哪个更牛逼?

    分享一份Java架构师学习资料

    640

    点击「阅读原文」一起搞技术,爽~

  • 相关阅读:
    使用Orachard与Bootstrap建站心得
    【02C语言】11函数的声明和定义
    【02C语言】09流程控制
    有趣的linux命令
    杭州哪家整容医院比较有威望?
    DDD:在基于关系数据库的领域,聚合的边界等于并发管理的边界。
    TOGAF架构开发方法(ADM)之业务架构阶段
    构建一个真实的应用电子商务SportsStore(八)
    Lucene分词组件盘古与mmseg4j评测
    .NET PDB文件到底是什么?
  • 原文地址:https://www.cnblogs.com/java-stack/p/11952261.html
Copyright © 2020-2023  润新知