垃圾回收概述
概念
这次我们主要关注的是黄色部分,内存的分配与回收
垃圾收集
垃圾收集,不是 Java 语言的伴生产物。早在 1960 年,第一门开始使用内存动态分配和垃圾收集技术的 Lisp 语言诞生。
关于垃圾收集有三个经典问题:
- 哪些内存需要回收?
- 什么时候回收?
- 如何回收?
垃圾收集机制是 Java 的招牌能力,极大地提高了开发效率。如今,垃圾收集几乎成为现代语言的标配,即使经过如此长时间的发展,Java 的垃圾收集机制仍然在不断的演进中,不同大小的设备、不同特征的应用场景,对垃圾收集提出了新的挑战,这当然也是面试的热点。
什么是垃圾?
在提到什么是垃圾之前,我们先看下面一张图
什么是垃圾(Garbage)呢?
- 垃圾是指在运行程序中没有任何指针指向的对象,这个对象就是需要被回收的垃圾。
- 英文:An object is considered garbage when it can no longer be reached from any pointer in the running program.
如果不及时对内存中的垃圾进行清理,那么,这些垃圾对象所占的内存空间会一直保留到应用程序的结束,被保留的空间无法被其它对象使用,甚至可能导致内存溢出。
磁盘碎片整理
机械硬盘需要进行磁盘整理,同时还有坏道
大厂面试题
蚂蚁金服
- 你知道哪几种垃圾回收器,各自的优缺点,重点讲一下 cms 和 G1?
- JVM GC 算法有哪些,目前的 JDK 版本采用什么回收算法?
- G1 回收器讲下回收过程 GC 是什么?为什么要有 GC?
- GC 的两种判定方法?CMS 收集器与 G1 收集器的特点
- 说一下 GC 算法,分代回收说下
- 垃圾收集策略和算法
天猫
- JVM GC 原理,JVM 怎么回收内存
- CMS 特点,垃圾回收算法有哪些?各自的优缺点,他们共同的缺点是什么?
滴滴
Java 的垃圾回收器都有哪些,说下 G1 的应用场景,平时你是如何搭配使用垃圾回收器的
京东
阿里
字节跳动
- 常见的垃圾回收器算法有哪些,各有什么优劣?
-
System.gc()
和Runtime.gc()
会做什么事情? - Java GC 机制?GC Roots 有哪些?
- Java 对象的回收方式,回收算法。
- CMS 和 G1 了解么,CMS 解决什么问题,说一下回收的过程。
- CMS 回收停顿了几次,为什么要停顿两次?
为什么需要GC
对于高级语言来说,一个基本认知是如果不进行垃圾回收,内存迟早都会被消耗完,因为不断地分配内存空间而不进行回收,就好像不停地生产生活垃圾而从来不打扫一样。
除了释放没用的对象,垃圾回收也可以清除内存里的记录碎片。碎片整理将所占用的堆内存移到堆的一端,以便 JVM 将整理出的内存分配给新的对象。
随着应用程序所应付的业务越来越庞大、复杂,用户越来越多,没有 GC 就不能保证应用程序的正常进行。而经常造成 STW 的 GC 又跟不上实际的需求,所以才会不断地尝试对 GC 进行优化。
早期垃圾回收
在早期的 C/C++ 时代,垃圾回收基本上是手工进行的。开发人员可以使用 new 关键字进行内存申请,并使用 delete 关键字进行内存释放。比如以下代码:
MibBridge *pBridge= new cmBaseGroupBridge();
//如果注册失败,使用Delete释放该对象所占内存区域
if(pBridge->Register(kDestroy)!=NO ERROR)
delete pBridge;
这种方式可以灵活控制内存释放的时间,但是会给开发人员带来频繁申请和释放内存的管理负担。倘若有一处内存区间由于程序员编码的问题忘记被回收,那么就会产生内存泄漏,垃圾对象永远无法被清除,随着系统运行时间的不断增长,垃圾对象所耗内存可能持续上升,直到出现内存溢出并造成应用程序崩溃。
MibBridge *pBridge=new cmBaseGroupBridge();
pBridge->Register(kDestroy);
现在,除了 Java 以外,C#、Python、Ruby等语言都使用了自动垃圾回收的思想,也是未来发展趋势,可以说这种自动化的内存分配和来及回收方式已经成为了线代开发语言必备的标准。
Java垃圾回收机制
优点
自动内存管理,无需开发人员手动参与内存的分配与回收,这样降低内存泄漏和内存溢出的风险
- 没有垃圾回收器,java 也会和 cpp 一样,各种悬垂指针,野指针,泄露问题让你头疼不已。
自动内存管理机制,将程序员从繁重的内存管理中释放出来,可以更专心地专注于业务开发
Oracle 官网关于垃圾回收的介绍
担忧
对于 Java 开发人员而言,自动内存管理就像是一个黑匣子,如果过度依赖于“自动”,那么这将会是一场灾难,最严重的就会弱化 Java 开发人员在程序出现内存溢出时定位问题和解决问题的能力。
此时,了解 JVM 的自动内存分配和内存回收原理就显得非常重要,只有在真正了解 JVM 是如何管理内存后,我们才能够在遇见 OutOfMemoryError 时,快速地根据错误异常日志定位问题和解决问题。
当需要排查各种内存溢出、内存泄漏问题时,当垃圾收集成为系统达到更高并发量的瓶颈时,我们就必须对这些“自动化”的技术实施必要的监控和调节。
GC主要关注的区域
垃圾收集器可以对年轻代回收,也可以对老年代回收,甚至是全栈和方法区的回收
- 其中,Java 堆是垃圾收集器的工作重点
从次数上讲:
- 频繁收集 Young 区
- 较少收集 Old 区
- 基本不收集 Perm 区(元空间)