第十四章 垃圾回收概述
1.什么是垃圾
-
垃圾是指在运行程序中没有任何指针指向的对象,这个对象就是需要被回收的垃圾。
-
如果不及时对内存中的垃圾进行清理,那么,这些垃圾对象所占的内存空间会一直保留到应用程序结束,被保留的空间无法被其它对象使用,甚至可能导致内存溢出。
2.为什么需要GC
-
对于高级语言来说,一个基本认知是如果不进行垃圾回收,内存迟早都会被消耗完,因为不断地分配内存空间而不进行回收,就好像不停地生产生活垃圾而从来不打扫一样。
-
除了释放没用的对象,垃圾回收也可以清除内存里的记录碎片。碎片整理将所占用的堆内存移到堆的一端,以便
JVM
将整理出的内存分配给新的对象。 -
随着应用程序所应付的业务越来越庞大、复杂,用户越来越多,没有
GC
就不能保证应用程序的正常运行。而经常造成STW
的GC
又跟不上实际的需求,所以才会不断地尝试对GC
的优化。
3.早期垃圾回收
-
在早期的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等语言都使用了自动垃圾回收的思想,也是未来发展趋势,可以说这种自动化的内存分配和来及回收方式已经成为了线代开发语言必备的标准。
4.Java
垃圾回收机制
-
自动内存管理,无需开发人员手动参与的分配与回收,这样降低内存泄漏和内存溢出的风险。
-
没有垃圾回收器,
java
也会和cpp
一样,各种悬垂指针、野指针、泄漏问题让你头疼不已。
弊端:
-
对于Java开发人员而言,自动内存管理就像是一个黑匣子,如果过度依赖于“自动”,那么这将会是一场灾难,最严重的就会弱化Java开发人员在程序出现内存溢出时定位问题和解决问题的能力。
-
此时,了解
JVM
的自动内存分配和内存回收原理就显得非常重要,只有在真正了解JVM
是如何管理内存后,我们才能在遇见outofMemoryError
时,快速地根据错误异常日志定位问题和解决问题。 -
当需要排查各种内存溢出、内存泄漏问题时,当垃圾收集成为系统达到更高并发量的瓶颈时,我们就必须对这些“自动化”的技术实施必要的监控和调节。
5.GC
主要关注的区域
GC
主要关注于方法区和堆中的垃圾回收
-
垃圾回收器可以对年轻代回收,也可以对老年代进行回收,甚至是全栈和方法区的回收。
-
其中,Java堆是垃圾收集器的工作重点
-
-
从次数上来讲
-
频繁收集Young区
-
较少收集Old区
-
-