浅谈Java 垃圾回收机制

谈谈你对Java 垃圾回收机制的理解,该如何回答呢?


1. 什么样的对象才被判定为垃圾?

简单的说就是,没有被其他对象引用的情况下的对象,可以判定为垃圾对象。


2. 怎么判断对象是否是垃圾?

垃圾回收主要有俩种算法:引用计数法可达性分析算法

引用计数法

概念:通过判断对象的引用计数,来决定对象是否可以被回收。每个对象实例都有一个引用计数器,被引用则+1,完成引用则-1,任何引用计数为0的对象,都可以被当成垃圾回收。

优点:执行效率高,程序受影响小。

缺点:无法检测出循环引用的情况,导致内存泄漏。

 

可达性分析算法

概念:通过判断对象的引用链是否可达,来决定对象是否可以被回收。通过一些名为(GC Roots (可以有多个))的对象作为起点,从这些节点开始向下搜索,搜索走过的路径被称为(Reference Chain) 引用链,当一个对象到GC Roots没有任何引用链相连时(即从GC Roots节点到该节点不可达),则证明该对象是不可用的。

优点:解决了引用计算法的无法循环引用的情况。

什么对象可以作为GC Roots 对象?

  • 虚拟机栈(栈帧中本地变量表)中引用的对象。
  • 方法区中类静态属性引用的对象。
  • 方法区中常量引用的对象。
  • 本地方法栈中JNI(Native方法)引用的对象。

3. 垃圾回收算法

垃圾回收算法主要有标记-清除算法、复制算法、标记整理算法、分代手机算法。

标记-清除算法(Mark And Sweep)

概念:标记-清除算法,分为俩个阶段,标记阶段、清除阶段;标记阶段的任务是标记出所有需要被回收的对象,清除阶段就是回收被标记的对象所占用的空间。

缺点:容易造成  碎片化(使空间不连续,容易产生内存溢出)。

复制算法(Copying)

概念:它将可用内存按容量划分为大小相等的两块,每次只使用其中的一块。当这一块的内存用完了,就将还存活着的对象复制到另外一块上面,然后再把已使用的内存空间一次清理掉,这样一来就不容易出现内存碎片的问题。

优点:解决了碎片化问题,顺序分配内存,简单高效,适用于对象存活率比较低的场景;

缺点:存活对象很多,那么Copying算法的效率将会大大降低。

标记-整理算法(Compacting)

概念:为了解决Copying算法的缺陷,充分利用内存空间,提出了Mark-Compact算法。该算法标记阶段和Mark-Sweep一样,但是在完成标记之后,它不是直接清理可回收对象,而是将存活对象都向一端移动,然后清理掉端边界以外的内存。

优点:避免内存的不连续性,不需要设置俩块内存互换,适用于存活率高的场景。

分代收集算法(Generational Collector)

概念:核心思想是根据对象存活的生命周期将内存划分为若干个不同的区域。一般情况下将堆区划分为老年代(Tenured Generation)和新生代(Young Generation),老年代的特点是每次垃圾收集时只有少量对象需要被回收,而新生代的特点是每次垃圾回收时都有大量的对象需要被回收,那么就可以根据不同代的特点采取最适合的收集算法。

目前大部分垃圾收集器对于新生代都采取复制算法,因为新生代中每次垃圾回收都要回收大部分对象,也就是说需要复制的操作次数较少,但是实际中并不是按照1:1的比例来划分新生代的空间的,一般来说是将新生代划分为一块较大的Eden空间和两块较小的Survivor空间,每次使用Eden空间和其中的一块Survivor空间,当进行回收时,将Eden和Survivor中还存活的对象复制到另一块Survivor空间中,然后清理掉Eden和刚才使用过的Survivor空间。而由于老年代的特点是每次回收都只回收少量对象,一般使用的是标记-整理算法(压缩法)


4. 常见的垃圾收集器

常见的垃圾收集器有Serial/Serial Old收集器、ParNew收集器、Parallel Scavenge收集器、Parallel Old 收集器、CMS 收集器、G1收集器等。

Serial/Serial Old收集器:是最基本最古老的收集器,它是一个单线程收集器,并且在它进行垃圾收集时,必须暂停所有用户线程。Serial收集器是针对新生代的收集器,采用的是Copying算法,Serial Old收集器是针对老年代的收集器,采用的是Mark-Compact算法。它的优点是实现简单高效,但是缺点是会给用户带来停顿。

ParNew收集器:是Serial收集器的多线程版本,使用多个线程进行垃圾收集。

Parallel Scavenge收集器:是一个新生代的多线程收集器(并行收集器),它在回收期间不需要暂停其他用户线程,其采用的是Copying算法,该收集器与前两个收集器有所不同,它主要是为了达到一个可控的吞吐量。

Parallel Old收集器:是Parallel Scavenge收集器的老年代版本(并行收集器),使用多线程和Mark-Compact算法。

CMS(Current Mark Sweep)收集器:是一种以获取最短回收停顿时间为目标的收集器,它是一种并发收集器,采用的是Mark-Sweep算法。

G1收集器:是当今收集器技术发展最前沿的成果,它是一款面向服务端应用的收集器,它能充分利用多CPU、多核环境。因此它是一款并行与并发收集器,并且它能建立可预测的停顿时间模型。

注:1. 主流的垃圾回收机制中,引用计数法很少被使用。原因:当对象之间相互指向时(即循环引用),两个对象的引用计数器的值都会+1,而由于两个对象时相互指向,所以引用不会失效,这样JVM就无法回收。

       2. 分代收集算法是目前大部分JVM垃圾回收集器采用的算法。

       3. 初步整理,有不完善的地方恳求指正,以便及时修改完善,欢迎各位吐槽,谢谢!

 

 

 

 

 

 

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值