引用计数器法(Reference Counting)

引用计数器的实现很简单,对于一个对象A,只要有任何一个对象引用了A,则A的引用计数器就加1,当引用失效时,引用计数器减1。只要对象A的引用计数器的值为0,则对象A不可能再被使用。

存在的问题:

l 无法处理循环引用,当对象A持有对象B的引用并且对象B持久对象A的引用,此时对象A和对象B的引用计数器都不为0。但是在系统中,却不存在任何第3个对象引用个A或B,此时A与b应该被回收,由于对象之间的相互引用导致垃圾回收期无法识别,引起内存泄漏。

l 引用计数器要求在每次引用产生和消除的时候,引用计数器需要加1或者减1对系统性能有一定影响。

由于以上原因JAVA并未采用此算法作为垃圾回收的算法

标记清除法(Mark-Sweep)

标记清除法将垃圾回收分为两个阶段:标记阶段和清除阶段。在标记阶段,首先通过根节点,标记所有从根节点开始的可达对象(通过根对象进行引用搜索,最终可以到达的对象)。因此未被标记的对象就是未被引用的垃圾对象。然后,在清除阶段,清除所有未被标记的对象。

存在的问题:

l 标记清除算法可能产生空间碎片,回收后空间不是连续的,在对象的对空间分配过程中,尤其是大对象的内存分配,不连续内存空间会造成性能损耗。

复制算法(Copying)

将原有的内存空间分为两块,每次只使用其中一块,在垃圾回收时,将正在使用的内存中存活的对象复制到未使用的内存快中,之后清除正在使用的内存块中的对象,交换两个内存的角色,完成垃圾回收。复制算法可以确保内存空间是没有碎片的,但是复制算法的代价却是将系统内存折半。

JAVA新生代垃圾回收

Java新生代的串行垃圾回收器中,使用了复制算法的思想。新生代分为eden、from、to三部分空间。其中from和to空间可以视为用于复制的两块大小相同、地位相等、且角色可互换的内存块。From和to空间也成为survivor空间(s1、s0),即幸存者空间,哟关于存放末被回收的对象。

在垃圾回收时,eden空间中的存活对象会被复制到未使用的survivor空间中(假设是to),正在使用的survivor空间(假设是from)中的年轻对象也会被复制到to空间中(大对象,或者老年对象会直接进入老年代,如果to空间占满,则对象也会直接进入老年代)。并且年龄加一。此时eden空间与from空间中的剩余对象则是垃圾对象,直接清空,to空间则存放此次回收后的存活对象。

标记压缩法(Mark-Compact)

复制算法的高效性是建立在存活对象少,垃圾对象多的前提下。但是老年代更常见的情况是大部分对象是存活对象,所以老年代采用了标记压缩算法。和标记清除算法一样,它也需要从根节点开始,对所有可达对象做一次标记。但之后它并不是简单的清除未标记对象,而是将所有的存活对象压缩在内存的一端,之后清理边界外所有的空间。标记压缩算法也可以成为标记清除压缩(MarkSweepCompact)

分代算法

将内存区间根据对象的特点分成几块,根据每块内存区间的特点,使用不同的回收算法,以提高回收的效率。JAVA虚拟机会将所有新建对象放入新生代内存区域(除超过PretenureSizeThreshold参数限制的大对象直接进入老年代),新生代的特点是新建对象很快会被回收,所以新生代采用复制算法较为合适。当一个对象经过几次垃圾回收后依然存活(每次年龄+1通过MaxTenuringThreshold参数可设置当年龄达到N时移入老年代),对象将会被放入老年代内存。在老年代中,几乎所有对象都是经过几次垃圾回收后依然存活的,因此,可以认为这些对象在一段时间内,甚至在应用程序的整个生命周期中,将是常驻内存的。根据分代思想,可以对老年代的回收使用与新生代不同的标记压缩算法。JVM为了支持新生代高频率的回收使用了一种叫做卡表(Card Table)的数据接口,卡表为一个比特集合,每一个比特位可以用来表示老年代某一区域是否有新生代的引用。这样新生代GC时只需要先扫描卡表,当卡表的标记为1时,才需要扫描相应区域的老年代对象。卡表中每一位表示老年代4k的空间。

分区算法

分区算法将整个堆划分成连续的不同小区间,每一个小区间都是独立使用独立回收。这种算法可以控制一次回收多少个小区域。

判断是否回收

可触及性可以包含以下3种状态

l 可触及:从根节点开始,可以到达这个对象。

l 可复活:对象的所有引用被释放,但对象有可能在finalize()方法中复活。

l 不可触及:对象的finalize()函数被调用,并且没有复活。Finalize只能被调用一次

以上三种状态中,只有对象不可触及才可以被回收。

对象的复活

package com.hl.gc;public class ReliveObj {public static ReliveObj obj;@Overrideprotected void finalize() throws Throwable {super.finalize();obj = this;}@Overridepublic String toString() {return "relive";}public static void main(String[] args) throws InterruptedException {obj = new ReliveObj();obj = null;System.gc();Thread.sleep(10000);String str;str = obj!=null?"可用":"不可用";System.out.println("Relive obj"+str);System.out.println("第二次 gc");obj = null;System.gc();Thread.sleep(10000);str = obj!=null?"可用":"不可用";System.out.println("Relive obj"+str);}
}

不推荐使用finalize方法释放资源,因为finalize方法可能发生引用外泄,在无意中无货对象,而且finalize是被系统调用的,调用时间是不明确的,因此不是一个好的释放资源方案,推荐使用try-catch-finallly进行释放资源。

引用和GC回收强度

Java 7之基础 - 强引用、弱引用、软引用、虚引用

l 强引用(StrongReference)

强引用是最普遍的引用,强引用可直接访问目标对象,强引用所指向的对象在任何时候都不会被系统回收,即使抛出OOM异常也不会回收强引用所指向的对象。强引用可能会造成内存溢出。

l 软引用(SoftReference

GC未必会回收软引用的对象,但是当内存不足时,软引用对象会被回收。所以软引用不会引起内存溢出(通常用来作为缓存使用)。

l 弱引用(WeakReference)

在GC回收时,只要发现弱引用,不管系统的堆使用情况如何,都会将对象回收。但是由于回收器的线程通常优先级较低,因此并不一定能很快地发现持有弱引用的对象。(通常也作为缓存使用)

l 虚引用(PhantomReference

一个持有虚引用的对象和没有引用几乎是一样的,随时可能被垃圾回收器回收。当试图通过虚引用的get()方法取得强引用时,总是会失败。并且,虚引用必须和引用队列一起使用,它的作用在于跟踪垃圾回收过程。当GC准备回收一个对象时,如果发现它还有虚引用,就会在回收对象后,将这个虚引用加入引用队列,已通知应用程序对象的回收情况。

垃圾回收的停顿现象:Stop-The-World

为了让垃圾回收器可以正常且高效的执行,大部分情况下,会要求系统进入一个停顿状态。停顿的目的是终止所有应用线程的执行,只有这样,系统中才不会产生新的垃圾,同时停顿保证了系统状态在某一个瞬间的一致性,有益于GC更好的标记垃圾对象。因此,在垃圾回收时,都会产生应用程序的停顿(STW),停顿产生时,整个应用程序会被卡死,没有任何响应。

©版权声明:本文为【翰林小院】(huhanlin.com)原创文章,转载时请注明出处!

转载于:https://www.cnblogs.com/hanlinhu/p/9487098.html

了解java虚拟机—垃圾回收算法(5)相关推荐

  1. JAVA虚拟机垃圾回收算法原理

    除了释放不再被引用的对象外,垃圾收集器还要处理堆碎块.新的对象分配了空间,不再被引用的对象被释放,所以堆内存的空闲位置介于活动的对象之间.请求分配新对象时可能不得不增大堆空间的大小,虽然可以使用的总空 ...

  2. JAVA虚拟机垃圾回收机制和JAVA排错三剑客

    一.Java虚拟机逻辑回收机制 1.Java垃圾回收器 Java垃圾回收器是Java虚拟机(JVM)的三个重要模块(另外两个是解释器和多线程机制)之一,为应用程序提供内存的自动分配(Memory Al ...

  3. Java虚拟机垃圾回收相关知识点全梳理(下)

    2019独角兽企业重金招聘Python工程师标准>>> 一.前言 上一篇文章<Java虚拟机垃圾回收相关知识点全梳理(上)>我整理分享了JVM运行时数据区域的划分,垃圾判 ...

  4. java jvm垃圾回收算法_深入理解JVM虚拟机2:JVM垃圾回收基本原理和算法

    本文转自互联网,侵删 本系列文章将整理到我在GitHub上的<Java面试指南>仓库,更多精彩内容请到我的仓库里查看 喜欢的话麻烦点下Star哈 文章将同步到我的个人博客: www.how ...

  5. Java虚拟机 —— 垃圾回收机制

    在Java虚拟机中,对象和数组的内存都是在堆中分配的,垃圾收集器主要回收的内存就是再堆内存中.如果在Java程序运行过程中,动态创建的对象或者数组没有及时得到回收,持续积累,最终堆内存就会被占满,导致 ...

  6. HotSpot 虚拟机垃圾回收算法实现

    作为使用范围最广的虚拟机之一HotSpot,必须对垃圾回收算法的执行效率有严格的考量,只有这样才能保证虚拟机高效运行 枚举根节点 从可达性分析中从 GC Roots 节点找引用链这个操作为例,可以作为 ...

  7. 老生常谈Java虚拟机垃圾回收机制(必看篇)

    垃圾收集 垃圾收集主要是针对堆和方法区进行. 程序计数器.虚拟机栈和本地方法栈这三个区域属于线程私有的,只存在于线程的生命周期内,线程结束之后也会消失,因此不需要对这三个区域进行垃圾回收. 判断一个对 ...

  8. Java虚拟机-垃圾回收简介

    一.如何判定对象为垃圾对象 -verbose:gc 打印垃圾回收简单信息参数 -xx:+PringDCDetail 打印垃圾回收的详细信息 引用计数法 引用计数算法很简单,它实际上是通过在对象头中分配 ...

  9. 深入理解Java虚拟机垃圾回收机制

    文章目录 什么是垃圾回收 哪些内存需要被回收?什么时候回收?如何回收? 哪些内存需要被回收?什么时候回收? 引用计数算法 可达性分析算法 如何回收?(垃圾收集算法) 标记-清除算法 复制算法 标记-整 ...

最新文章

  1. python精进之路 -- open函数
  2. cordova在IOS初次运行,loading加载很久的解决办法
  3. 一文看透 Redis 分布式锁进化史(解读 + 缺陷分析)
  4. AutoScaling 生命周期挂钩功能
  5. [BAT][JAVA]定时任务之-Quartz使用篇(通过这个配置可以知道在做Quartz的时候需要的jar文件/Cron表达式使用语法/常用Cron表达式)
  6. MYSQL 更改数据库data存储目录 创建用户 创建权限 设置远程访问的权限.
  7. 【贪心】国王游戏(ybtoj 贪心-1-4)
  8. JS数组的相关操作(循环、查找、过滤、排序等)
  9. 解决64位进程调用32位库文件报错问题
  10. linux下编译upx ucl
  11. word中导出高分辨率pdf
  12. 8.2 自制操作系统: risc-v Machine寄存器说明mstatus和mstatush
  13. [NIPS 18] Stacked Semantics-Guided Attention Model for Fine-Grained Zero-Shot Learning
  14. c语言已知斜率 求倾角,直线与方程(一)倾斜角与斜率
  15. esp_easy固件配合domoticz用于智能家居温湿度监控
  16. JS开发HTML5游戏《神奇的六边形》(四)
  17. GNSS数据下载脚本(Perl+Python)
  18. 多元回归分析(分类与运用)
  19. vue实现头部吸顶描点
  20. 掌握合影拍摄技巧 拍出不一样的全家福

热门文章

  1. piwik的安装与配置
  2. Linux工具之curl与wget高级使用
  3. (十二)Flask 学习 —— 换装
  4. 人生的意义—我们为什么活着?
  5. centos中用MySQL创建新表_CentOS下使用Shell批量创建数据库表
  6. 数字信号处理基础----FM的调制与解调
  7. Vivado 中IP报严重警告Could not find module的解决办法
  8. 通信中dBFS、dBm、dBV、dBW、0dB、-3dB的定义
  9. jittor和pytorch生成网络对比之stargan
  10. jittor和pytorch生成网络对比之cogan