JVM学习笔记二之GC

GC即垃圾回收,在C++中垃圾回收由程序员自己来做,例如可以用free和delete来回收对象。而在Java中,JVM替程序员来执行垃圾回收的工作,下面看看GC的详细原理和执行过程。

1.对象已死?

1.1 引用计数法

Java GC不采用引用计数法是因为无法解决对象互相引用导致无法回收的问题。

1.2 可达性分析法

从GC ROOT开始搜索,搜索不到的并且经过第一次标记、清理后未复活的对象。

GC ROOT包括:

  • 虚拟机栈中引用的对象
  • 方法区中静态属性引用的对象
  • 方法区中常量引用的对象
  • Native方法引用的对象

1.3 四种引用

强引用

  1. Object object = new Object();

只要强引用存在,就不会被回收。

软引用

描述一些还有用但是非必需的对象,当系统将要发生OOM之前,会对软引用的对象进行二次回收。

可以用SoftReference来实现软引用。

弱引用

也是用来描述一些非必需的对象,只能生存到下次垃圾回收之前,可以用WeakReference来实现。

虚引用

引用强度最弱的一种引用,可以用PhantomReference来实现虚引用。

2.垃圾回收算法

2.1 标记-清除算法

两个不足:

  • 标记和清除两个阶段效率都很低
  • 会造成大量的内存碎片,而为大对象分配内存需要大段的连续内存

2.2 复制算法

将内存分为两块同等大小的区域,一块用完时把上面存活的对象复制到另一块内存上,然后对这块内存进行集体回收。

这种做法的坏处是空间利用率太低。

2.3 标记-整理算法

标记整理算法是让所有存活对象都向一端移动,然后直接清理掉端外的所有内存。

2.4 分代收集算法

在现在的主流JVM中,采用的都是分代回收算法,对于经常需要回收的新生代,采用的是复制算法,而对于存活率高的老年代,采用的是标记清除和标记整理算法。

3.垃圾收集器

3.1 Serial收集器

单线程收集器,在进行垃圾收集时会停止所有其他线程,即stop-the-world。

  1. -XX:+UseSerialGC

3.2 ParNew收集器

ParNew是Serial的多线程版本,但是还是会有stop-the-world的问题。

  1. -XX:+UseParNewGC

3.3 Parallel Scavenge收集器

类似ParNew收集器,但是更关注系统的吞吐量,如果更注重吞吐量应该使用Parallel Scavenge收集器。

3.4 Serial Old收集器

Serial收集器的老年代版本,主要进行老年代的收集。

3.5 Parallel Old收集器

SerialOld的多线程版本。

3.6 CMS(Concurrent Mark Sweep)收集器

  1. -XX:+UseConcMarkSweepGC

收集器是一种以获取最短回收停顿时间为目标的收集器。

但是有以下缺点:

  • 由于采用多线程占用了CPU资源,可能导致系统性能下降
  • 无法处理浮动垃圾:由于GC线程与用户线程同时运行,导致GC过程中还可能有垃圾产生
  • 由于CMS是标记-清除算法,所以会造成内存碎片的产生

3.7 G1收集器

有如下几点特点:

  • 并发与并行:通过利用多CPU、多核来减少stop-the-world的时间
  • 分代回收:可以独当一面,不需要其他收集器的配合
  • 空间整合:从整体上看是标记-整理算法,不同于CMS的标记-清除算法,可以有效地减少内存碎片的产生
  • 可预测的停顿:采用Region进行分区,可以把一个Region分配到任意一个区域,然后会选择清理收益最高的一个Region(相比于直接清理年轻代或老年代来说清理的粒度更细,从而就导致效率提升)。

垃圾回收过程

Eden:Survivor From:Survivor TO = 8 : 1 : 1

为什么Survivor区域要分为两个:让对象在之间来回复制,从而计算对象存活的年龄。

http://ifeve.com/jvm-yong-generation/

什么时候会触发GC

由于GC是分代回收,所以这个问题分为什么时候触发Minor GC(年轻代)、Major GC(老年代)和Full GC(年轻代+老年代):

  • Minor GC:当Eden区域满了的时候,用户又需要创建对象,需要在堆中的年轻代分配内存,此时就会触发Minor GC。Minor GC主要采用复制收集算法,由于年轻代中的对象都很小,所以GC时间很短,经过Minor GC后没有被回收的对象会被移动到Survivor区域,并且年龄加一,当年龄超过阈值时会被移动到老年代中。

  • Major GC:当对象从年轻代移动到老年代之前,会检测老年代中剩余的空间是否足够装得下这些对象,所以当老年代内存不足时会触发Major GC。Major GC主要采用的标记-整理算法(CMS)。

  • Full GC:除了直接调用System.gc()之外,还有以下四种情况:

    • 老年代内存不足
    • 永生代内存不足
    • CMS GC时出现promotion failed和concurrent mode failure
    • 统计得到的Minor GC晋升到旧生代的平均大小大于旧生代的剩余空间

【Java虚拟机】JVM学习笔记之GC的更多相关文章

  1. java虚拟机JVM学习笔记-基础知识

    最近使用开发的过程中出现了一个小问题,顺便记录一下原因和方法--java虚拟机 媒介:JVM是每一位从事Java开发工程师必须翻越的一座大山! JVM(Java Virtual Machine)JRE ...

  2. java之jvm学习笔记十三(jvm基本结构)

    java之jvm学习笔记十三(jvm基本结构) 这一节,主要来学习jvm的基本结构,也就是概述.说是概述,内容很多,而且概念量也很大,不过关于概念方面,你不用担心,我完全有信心,让概念在你的脑子里变成 ...

  3. 《深入理解Java虚拟机》学习笔记

    <深入理解Java虚拟机>学习笔记 一.走近Java JDK(Java Development Kit):包含Java程序设计语言,Java虚拟机,JavaAPI,是用于支持 Java 程 ...

  4. java之jvm学习笔记六-十二(实践写自己的安全管理器)(jar包的代码认证和签名) (实践对jar包的代码签名) (策略文件)(策略和保护域) (访问控制器) (访问控制器的栈校验机制) (jvm基本结构)

    java之jvm学习笔记六(实践写自己的安全管理器) 安全管理器SecurityManager里设计的内容实在是非常的庞大,它的核心方法就是checkPerssiom这个方法里又调用 AccessCo ...

  5. java之jvm学习笔记三(Class文件检验器)

    java之jvm学习笔记三(Class文件检验器) 前面的学习我们知道了class文件被类装载器所装载,但是在装载class文件之前或之后,class文件实际上还需要被校验,这就是今天的学习主题,cl ...

  6. java之jvm学习笔记四(安全管理器)

    java之jvm学习笔记四(安全管理器) 前面已经简述了java的安全模型的两个组成部分(类装载器,class文件校验器),接下来学习的是java安全模型的另外一个重要组成部分安全管理器. 安全管理器 ...

  7. java之jvm学习笔记二(类装载器的体系结构)

    java的class只在需要的时候才内转载入内存,并由java虚拟机的执行引擎来执行,而执行引擎从总的来说主要的执行方式分为四种, 第一种,一次性解释代码,也就是当字节码转载到内存后,每次需要都会重新 ...

  8. Java四种引用--《深入理解Java虚拟机》学习笔记及个人理解(四)

    Java四种引用--<深入理解Java虚拟机>学习笔记及个人理解(四) 书上P65. StrongReference(强引用) 类似Object obj = new Object() 这类 ...

  9. Java虚拟机内存溢出异常--《深入理解Java虚拟机》学习笔记及个人理解(三)

    Java虚拟机内存溢出异常--<深入理解Java虚拟机>学习笔记及个人理解(三) 书上P39 1. 堆内存溢出 不断地创建对象, 而且保证创建的这些对象不会被回收即可(让GC Root可达 ...

  10. 【Java】「深入理解Java虚拟机」学习笔记(1) - Java语言发展趋势

    0.前言 从这篇随笔开始记录Java虚拟机的内容,以前只是对Java的应用,聚焦的是业务,了解的只是语言层面,现在想深入学习一下. 对JVM的学习肯定不是看一遍书就能掌握的,在今后的学习和实践中如果有 ...

随机推荐

  1. 洛谷P1124 文件压缩

    https://www.luogu.org/problem/show?pid=1124 题目背景 提高文件的压缩率一直是人们追求的目标.近几年有人提出了这样一种算法,它虽然只是单纯地对文件进行重排,本 ...

  2. Elasticsearch技术解析与实战(六)Elasticsearch并发

    乐观锁与悲观锁 图示的冲突过程,其实就是es的并发冲突问题,会导致数据不准确 当并发操作es的线程越多,或者读取一份数据,供用户查询和操作的时间越长,在这段时间里,如果数据被其他用户修改,那么我们拿到 ...

  3. [linux]安装code::blocks

    1.安装基本编译环境 $sudo apt-get install build-essential $sudo apt-get install gdb 2.安装codeblock $sudo apt-g ...

  4. 第七周 ch04 课下测试补交

    2017-2018-1 20155335 <信息安全系统设计基础>第7周 课下测试博客 本人不慎忘记去交dao'zhi 测试题目: SEQ+对SEQ的改变有() A . PC的计算挪到取指 ...

  5. Let's Encrypt 免费通配 https 签名证书 安装方法2 ,安卓签名无法认证!

    Let's Encrypt 免费通配 https 签名证书 安装方法 按照上文 配置完毕后你会发现 在pc浏览器中正常访问,在手机浏览器中无法认证 你只需要安装一个或多个中级证书 1.查看Nginx ...

  6. 垂直水平居中--css3

    在移动前端制作中,很多新的css3特性能够帮助我们更好的制作.例如这个垂直水平居中问题,就有一个简单的代码可以解决: 利用CSS3的transform:translate .center{ width ...

  7. hihoCoder 1174 : 拓扑排序·一

    题目链接:http://hihocoder.com/problemset/problem/1174 题目是中文题面我就不说题意了,要看题面的请点击上方链接~ 代码实现如下: #include < ...

  8. Kali设置代理

    原文:Kali-linux设置ProxyChains ProxyChains是Linux和其他Unices下的代理工具.它可以使任何程序通过代理上网,允许TCP和DNS通过代理隧道,支持HTTP.SO ...

  9. android ViewPager之OnPageChangeListener接口

    项目中在使用ViewPager的时候,一般都要在界面滑动的时候做一些事情,android中有个专门的状态回调接口OnPageChangeListener. /** * Callback interfa ...

  10. (4)剑指Offer之链表相关编程题

    一 链表中倒数第k个节点 题目描述: 输入一个链表,输出该链表中倒数第k个结点 问题分析: 一句话概括: 两个指针一个指针p1先开始跑,指针p1跑到k-1个节点后,另一个节点p2开始跑,当p1跑到最后 ...