Java虚拟机的JVM垃圾回收机制

1.JVM内存空间

 

  JVM堆(Heap)= 新生代(Young) + 旧生代(Tenured)

 分区作用: 

新创建的对象通常先将其分配在新生代中,在新生代中经过若干次GC之后仍未释放的对象,再将它移动到旧生代。为了让内存回收更高效(GC会暂停JVM中的应用),Sun JDK在1.2开始对堆采用了分代管理的方式。在分配对象遇到内存不足时,先对新生代进行GC(Young GC);当新生代GC之后仍无法满足内存空间分配需求时, 才会对整个堆空间以及方法区进行GC (Full GC)

 相关参数:

-Xms                       -- 设置堆内存初始大小

 -Xmx                       -- 设置堆内存最大值

-XX:MaxTenuringThreshold  -- 设置对象在新生代中存活的次数

-XX:PretenureSizeThreshold  -- 设置超过指定大小的大对象直接分配在旧生代中

 注意点:当新生代设置得太小时,也可能引发大对象直接分配到旧生代中。

 

  新生代(Young)= Eden区 + Survivor区

 分区作用:

Eden区为对象通常最初分配到的地方,Survivor区分为S0和S1两块大小相等的区域。JVM进行Minor GC时,将Eden中还存活的对象拷贝到Survivor区中,还会将Survivor区中还存活的对象拷贝到Tenured区中。在这种GC模式下,JVM为了提升GC效率, 将Survivor区分为S0和S1,这样就可以将对象回收和对象晋升分离开来。

 相关参数:

-Xmn             -- 设置新生代内存大小。

-XX:SurvivorRatio  -- 设置Eden与Survivor空间的大小比例

 注意点: 图中Virtual部分表示可伸缩的内存空间,当用-Xms在指定堆的初始大小为128m,通过-Xmx指定堆最大为256m时,JVM会根据内存情况在128m与256m之间伸缩。为了避免JVM进行这些伸缩消耗性能,对于能够提供稳定内存空间的用作服务器的JVM,通常将-Xms和-Xmx设置为相等。

  方法区(Perm)

 分区作用:

也被成为持久代,用来存放JVM加载的类型信息。包括: 类型基本信息,常量池,字段信息,方法信息,类变量,指向ClassLoader的引用,Class类的引用,方法表等。方法区是全局共享的,在一定条件下也会被GC。

 相关参数:

-XX:PermSize        --设置Perm区的初始大小

-XX:MaxPermSize    --设置Perm区的最大值

 

  JVM方法栈(注意:不是分区)

 作用: 

JVM方法栈为JVM线程私有内存,当方法运行完毕后,其对应的栈帧内存会自动释放

 相关参数:

-Xss                  --设置方法栈的最大值

 TLAB:

JVM所占用的主要内存都是从堆空间分配的,堆是所有线程共享的,因此在堆上分配内存需要加锁,Sun JDK为提升效率,会为每个新建的线程在Eden上分配一块独立的空间由该线程独享,这块空间称为TLAB(Thread Local Allocation Buffer)。其大小由JVM根据运行情况计算得到,也可通过参数-XX:TLABWasteTargetPercent来设置TLAB可占用的Eden空间的百分比,默认值为1%。在TLAB上分配内存不需要加锁,因此JVM在给线程中的对象分配内存时会尽量在TLAB上分配。如果对象过大或TLAB用完,则仍然在堆上进行分配。

2.Sun JDK GC收集器

收集器主要分为引用计数器和跟踪收集器两种,Sun JDK中采用跟踪收集器作为GC实现策略。

  跟踪收集器简介

跟踪收集器采用集中式的管理方式,全局记录数据的引用状态。触发执行时需要从根节点来扫描对象的引用关系,可能会造成应用程序暂停。主要有三种实现算法:复制(Copying) 、 标记-清除(Mark-Sweep) 、 标记-压缩(Mark-Compact) 。下文将简单介绍这三种算法的过程,有助于后续的GC策略理解和分析。

  复制(Copying)

 算法:复制采用的方式为从根集合扫描出存活的对象,并将找到的存活对象复制到一块新的完全未使用的空间中。 过程:注意,红叉为不存活的对象所占用内存空间

  标记-清除(Mark-Sweep)

 算法:标记-清除采用的方式为从根集合开始扫描,对存活的对象进行标记,标记完毕后,再扫描整个空间中未标记的对象,并进行回收。 过程:

ü   优缺点:在空间中存活对象较多的情况下较为高效,但由于该算法为直接回收不存活对象所占用的内存,因此会造成内存碎片。

 

标记-压缩(Mark-Compact)

ü   算法:标记阶段与“标记-清除”算法相同,但在清除阶段有所不同。在回收不存活对象所占用的内存空间后,会将其他所有存活对象都往左端空闲的空间进行移动,并更新引用其对象指针。

ü   过程:

 

ü   优缺点:在“标记-清除”的基础上还需要进行对象移动,成本相对较高,好处则是不产生内存碎片。

  

 

3.  Sun JDK GC策略

 

图4 Sun JDK中可用的GC方式

 

基于上一小节讲解的跟踪收集器算法,Sun JDK在新生代和老生代进行了不同的算法实现,形成了上图中的GC方式分布。本小节将具体介绍新生代和老生带的GC策略及组合方式。

 

  新生代 – 串行GC(Serial Copying)

 算法:复制(Copy)

 过程:

 1. 扫描出新生代中存活的对象;

 2. Minor GC将存活的对象复制到做为To Space的S0/S1区;

 3. 之前做为To Space/From Spache的S0/S1区对换角色;

 4. 经历过几次Minor GC仍然存活的对象,放入老生代。

  新生代 – 并行回收GC(Parallel Scavenge)

 算法:复制(Copy)

 过程:在扫描和复制时均采用多线程方式进行(如下图),并且并行回收GC为大的新生代回收做了很多优化(可以自行扩展阅读相关资料)。

 优势:在多CPU的机器上其GC耗时会比串行方式短,适合多CPU、对暂停时间要求较短的应用。 配置方式:本身是Server级别多CPU机器上的默认GC方式,也可以通过-XX:+UseParallelGC来指定,并且可以采用-XX:ParallelGCThread来指定线程数。

  新生代 – 并行GC(ParNew)

 算法:复制(Copy) 过程:与并行回收GC(Parallel Scavenge)的区别在于并行GC(ParNew)必须配合老生代使用CMS GC。原因是CMS GC在进行老生代GC时,有些过程是并发执行的。如果此时发生了Minor GC,需要进行相应处理,而并行回收GC(Parallel Scavenge)是没有做这些处理的。也正是如此,ParNew GC不可与并行的老生代GC同时使用。 配置方式:在配置为CMS GC的情况下,新生代默认使用并行GC(ParNew)方式,也可以通过-XX:+UseParNewGC来指定。

  老生代 – 串行GC(Serial MSC)

 算法:Mark-Sweep-Compact,该算法结合Mark-Sweep和Mark-Compact做了一些改进。 过程: 1. 扫描出老生代中存活的对象,进行标识; 2. 遍历整个老生代和持久代内存空间,找出未被标识的对象,并回收其内存; 3. 进行滑动压缩(Sliding Compaction),将存活对象向老生代空间的开始处滑动,最终留出一块连续的到结尾的内存空间。 优缺点:串行执行的过程中为单线程,需要暂停应用并耗时较长。 配置方式:是client模式默认采用的GC方式,也可以通过-XX:UseSerialGC进行指定。

  老生代 – 并行GC(Parallel Mark Sweep、Parallel Compacting)

 算法:Mark -Compact 过程: 1. 将老生代划分为并行线程个数的区域(regions); 2. 并行进行存活对象扫描和标记; 3. 单线程对各区域进行扫描,标记需要压缩移动的区域; 4. 并行进行对象移动和区域不存活对象的回收。 优缺点:多线程同时操作以及dense prefix优化,会缩短应用暂停时间。但由于老生代较大,在扫描和标识对象上需要花费较长时间。 配置方式:通过-XX:+UseParallelGC来指定使用Parallel Mark Sweep;通过-XX:UseParallelOldGC来指定使用Parallel Compacting。

  老生代 – 并发GC(CMS:Concurrent Mark-Sweep GC)

 算法:Mark –Sweep 过程: 1. 第一次标记(Initial Marking):暂停整个应用,扫描从根集合点到老生代中可直接访问到的对象,并进行标记; 2. 并发标记(Concurrent Marking):恢复所有应用的线程,同时开始并发对之前标记过的对象进行轮循,以标记这些对象可访问的对象; 3. 重新标记(Remark):暂停整个应用,扫描在第二步中被改变引用关系或新创建的对象,并进行标记; 4. 并发收集(Concurrent Sweeping):恢复所有应用的线程,将没有标记的对象进行单线程回收。针对内存碎片,CMS会尽量将相邻的块重新组装成一个块。

 

 优缺点:如上图,优点是只有在第一次标记和重新标记阶段需要暂停整个应用,所以能够做到影响应用响应时间很短。缺点是并发标记和并发收集阶段CMS会与应用线程争用CPU资源(用增量CMS模式可以缓解),并且容易产生内存碎片,free-list机制会导致Minor GC效率下降。

 配置方法:通过-XX:UseConcMarkSweepGC来启动老生代CMS GC;通过-XX:+UseCMSCompactAtFullCollection来启动内存碎片整理功能(整理也会暂停应用)。

 

  4.Sun JDK GC默认策略及组合策略

ü   Clinet、Server模式默认GC策略

新生代GC方式


旧生代和持久代GC方式


Client


串行GC


串行GC


Server


并行回收GC


Parallel Mark Sweep GC

 

 

ü   Sun JDK GC组合方式

新生代GC


旧生代和持久代GC


-XX:+UseSerialGC


串行GC


串行GC


-XX:+UseParallelGC


并行回收GC


Parallel Mark Sweep GC


-XX:+UseConcMarkSweepGC


并行GC


并发GC

当出现Concurrent Mode Failure时采用串行GC


-XX:+UseParNewGC


并行GC


串行GC


-XX:+UseParallelOldGC


并行回收GC


Parallel Mark Conpact


-XX:+UseConcMarkSweepGC

-XX:-UseParNewGC


串行GC


并发GC

当出现Concurrent Mode Failure或Promotion Failed时采用串行GC


不支持的组合方式


1.-XX:+UseParNewGC -XX:+UseParallelOldGC

2.-XX:+UseParNewGC -XX:+UseSerialGC

 

特别说明:尊重作者的劳动成果,转载请注明出处哦~~~http://blog.yemou.net/article/query/info/tytfjhfascvhzxcytp43

时间: 2024-10-29 22:05:40

Java虚拟机的JVM垃圾回收机制的相关文章

JVM内存管理、JVM垃圾回收机制、新生代、老年代以及永久代

如果大家想深入的了解JVM,可以读读周志明<深入理解Java虚拟机:JVM高级特性与最佳实践>      需要掌握的东西,包括以下内容.判断对象存活还是死亡的算法(引用计数算法.可达性分析算法).常见的垃圾收集算法(复制算法.分代收集算法等以及这些算法适用于什么代)以及常见的垃圾收集器的特点(这些收集器适用于什么年代的内存收集).            JVM运行时数据区由程序计数器.堆.虚拟机栈.本地方法栈.方法区部分组成,结构图如下所示.      JVM内存结构由程序计数器.堆.栈.本地

JAVA虚拟机内存分配与回收机制

Java虚拟机(Java Virtual Machine) 简称JVM Java虚拟机是一个想象中的机器,在实际的计算机上通过软件模拟来实现.Java虚拟机有自己想象中的硬件,如处理器.堆栈.寄存器等,还具有相应的指令系统. Java把内存划分成两种:一种是栈内存,一种是堆内存. 在函数中定义的一些基本类型的变量和对象的引用变量都在函数的栈内存中分配. 当在一段代码块定义一个变量时,Java就在栈中为这个变量分配内存空间,当超过变量的作用域后,Java会自动释放掉为该变量所分配的内存空间,该内存

从JVM的内存管理角度分析Java的GC垃圾回收机制_java

一个优秀的Java程序员必须了解GC的工作原理.如何优化GC的性能.如何与GC进行有限的交互,因为有一些应用程序对性能要求较高,例如嵌入式系统.实时系统等,只有全面提升内存的管理效率 ,才能提高整个应用程序的性能.本篇文章首先简单介绍GC的工作原理之后,然后再对GC的几个关键问题进行深入探讨,最后提出一些Java程序设计建议,从GC角度提高Java程序的性能.    GC的基本原理    Java的内存管理实际上就是对象的管理,其中包括对象的分配和释放.     对于程序员来说,分配对象使用ne

析JAVA之垃圾回收机制

本文为2010年编写,所以有很多看法不是很准确,有一定的参考价值,如需要更加深入细节,请参看,2012年编写的关于JVM的文章: 认识JVM--第一篇-对象生成&回收算法 认识JVM--第二篇-java对象内存模型 JVM第三篇(简单demo) 系统架构-性能篇章1(应用系统性能2-OOM&参数配置) 相继的还会有更多的java深入的知识和机制. 对于JAVA编程和很多类似C.C++语言有一个巨大区别就是内存不需要自己去free或者delete,而是由JVM垃圾回收机制去完成的.对于这个过

Android垃圾回收机制解决内存泄露问题_Android

在android编码中,会有一些简便的写法和编码习惯,会导致我们的代码有很多内存泄露的问题,在这里做一个已知错误的总结:1.编写单例的时候常出现的错误. 错误方式: public class Foo{ private static Foo foo; private Context mContext; private Foo(Context mContext){ this.mContext = mContext; } // 普通单例,非线程安全 public static Foo getInsta

.Net 垃圾回收机制详细介绍_实用技巧

析构函数 析构函数不能有修饰符,如public.不能接受任何参数. 编译器自动将一个析构函数转换成对Object.Finalize方法的一个override版,如下. class Test { protected override void Finalize() { try {-} finally { base.Finalize(); } } } 垃圾回收器 .NET垃圾回收器会保证: l  每个对象都会被摧毁,它的析构函数一定会被运行.当一个程序结束后,所有对象都会被销毁. l  每个对象只被

解读PHP中的垃圾回收机制_php技巧

PHP的基本GC概念PHP语言同其他语言一样,具有垃圾回收机制.那么今天我们要为大家讲解的内容就是关于PHP垃圾回收机制的相关问题.希望对大家有所帮助.PHP strtotime应用经验之谈PHP memory_get_usage()管理内存PHP unset全局变量运用问题详解PHP unset()函数销毁变量教你快速实现PHP全站权限验证一.PHP 垃圾回收机制(Garbage Collector 简称GC) 在PHP中,没有任何变量指向这个对象时,这个对象就成为垃圾.PHP会将其在内存中销

[JVM]成为JavaGC专家(1)—深入浅出Java垃圾回收机制

对于Java开发人员来说,了解垃圾回收机制(GC)有哪些好处呢?首先可以满足作为一名软件工程师的求知欲,其次,深入了解GC如何工作可以帮你写出更好的Java应用. 这仅仅代表我个人的意见,但我坚信一个精通GC的人往往是一个好的Java开发者.如果你对GC的处理过程感兴趣,说明你已经具备较大规模应用的开发经验.如果你曾经想过如何正确的选择GC算法,那意味着你已经完全理解你所开发的应用的特点.当然,我们不能以偏概全,这不能作为评价一个好的开发人员的共通标准.但是,我要说的是,深入理解GC是成为一名伟

细述 Java垃圾回收机制→Java Garbage Collection Introduction

计划写一个介绍Java垃圾回收基础的系列文章,共分四部分: Java垃圾回收简介 Java垃圾回收器是如何工作的? 各种类型的Java垃圾回收器 Java垃圾回收的监控和分析 本文是这个系列的第一篇文章,这篇文章将会介绍一些基本术语,如:JDK,JVM,JRE,HotSpot VM,以及理解JVM的架构和Java堆内存结构.在开始学习Java垃圾回收机制之前确实有必要了解一下这些基本东西. 关键的Java术语 Java API–一个帮助程序员创建Java应用的打包好的库集合 Java Devel