Java垃圾回收机制
jvm的gc原理

jvm的gc原理JVM的GC原理一、概述JVM(Java虚拟机)是Java程序运行的环境,其中最重要的组成部分之一就是垃圾回收(Garbage Collection,简称GC)机制。
GC的作用是自动管理程序中的内存,及时释放不再使用的对象,以避免内存泄漏和内存溢出的问题。
本文将对JVM的GC原理进行详细介绍。
二、垃圾回收算法1. 标记-清除算法标记-清除算法是最基本的垃圾回收算法之一。
它的过程分为两个阶段:标记阶段和清除阶段。
在标记阶段,GC会从根节点(一般是程序中的静态变量和栈中的引用)开始,递归地遍历对象图,标记出所有被引用的对象。
在清除阶段,GC会遍历整个堆,清除所有未被标记的对象。
2. 复制算法复制算法是针对标记-清除算法的改进。
它将堆分为两个区域,每次只使用其中一个区域。
当一个区域的对象被标记后,将其复制到另一个区域中,然后清除原来的区域。
这样可以解决碎片问题,但是需要额外的空间来存储复制的对象。
3. 标记-整理算法标记-整理算法是对标记-清除算法的改进。
它的过程与标记-清除算法类似,但是在清除阶段,标记-整理算法会将存活的对象向一端移动,然后清除边界外的所有对象。
这样可以解决碎片问题,并且不需要额外的空间。
4. 分代算法分代算法是针对对象的生命周期不同而提出的。
一般来说,对象的生命周期可以分为年轻代和老年代。
年轻代中的对象生命周期较短,老年代中的对象生命周期较长。
分代算法将堆分为年轻代和老年代两个区域,分别采用不同的垃圾回收算法。
年轻代一般使用复制算法,老年代一般使用标记-清除算法或标记-整理算法。
三、GC的执行过程1. 初始标记初始标记阶段是GC的第一步,它的目的是标记出所有的根对象,并且停止所有的应用线程。
这个过程是短暂的,因为只需要标记出与根对象直接关联的对象。
2. 并发标记并发标记阶段是GC的核心步骤,它的目的是通过并发执行来标记出所有的存活对象。
在这个阶段,GC会遍历整个堆,标记出与根对象直接或间接关联的存活对象。
javaGC垃圾回收机制G1、CMS

javaGC垃圾回收机制G1、CMSCMS(Concurrent Mark-Sweep)是以牺牲吞吐量为代价来获得最短回收停顿时间。
对于要求服务器响应速度的应⽤上,这种垃圾回收器⾮常适合。
在启动JVM参数加上-XX:+UseConcMarkSweepGC ,这个参数表⽰对于⽼年代的回收采⽤CMS。
CMS采⽤的基础算法是:标记—清除。
使⽤场景:1、应⽤程序对停顿⽐较敏感,并且在应⽤程序运⾏的时候可以提供更⼤的内存和更多的CPU2、在JVM中,有相对较多存活时间较长的对象(⽼年代⽐较⼤)会更适合使⽤CMS。
为解决CMS算法产⽣空间碎⽚和其它⼀系列的问题缺陷,HotSpot提供了另外⼀种垃圾回收策略,G1(Garbage First)算法,通过参数-XX:+UseG1GC来启⽤,该算法在JDK 7u4版本被正式推出,G1垃圾收集算法主要应⽤在多CPU⼤内存的服务中,在满⾜⾼吞吐量的同时,竟可能的满⾜垃圾回收时的暂停时间,下⾯是官⽅介绍:The Garbage-First (G1) collector is a server-style garbage collector, targeted for multi-processor machines with large memories.It meets garbage collection (GC) pause time goals with a high probability, while achieving high throughput. The G1 garbagecollector is fully supported in Oracle JDK 7 update 4 and later releases. The G1 collector is designed for applications that:Can operate concurrently with applications threads like the CMS collector.Compact free space without lengthy GC induced pause times.Need more predictable GC pause durations.Do not want to sacrifice a lot of throughput performance.Do not require a much larger Java heap.G1采⽤了另外⼀种完全不同的⽅式组织堆内存,堆内存被划分为多个⼤⼩相等的内存块(Region),每个Region是逻辑连续的⼀段内存,G1中提供了三种模式垃圾回收模式,young gc、mixed gc 和 full gc,在不同的条件下被触发。
java自动垃圾回收机制

java⾃动垃圾回收机制前⾔:相⽐C++,java做的⼀⼤改进是将复杂的内存管理抽离出来交给jvm去处理,让码农不再时刻盯着内存泄漏的问题,可以更专注于业务逻辑的开发。
java的GC机制是和其内存模型相关联的,⽽GC的核⼼内存区域是内存中的堆区。
java堆区按对象的存活时间被分为了年轻代(eden区+s0区+s1区)和⽼年代(tentired区),java堆的按代区分其实是为了其垃圾回收的分代收集机制打开了⽅便之门。
java的GC收集器会在不同的分代上使⽤不同的垃圾收集策略。
GC其实主要需要解决两个问题:哪些是垃圾?如何清理垃圾?在解决这两个问题上涉及到下⾯的⽅法论:1.垃圾对象判定⽅法引⽤计数法:在C++的智能指针中使⽤了这种⽅式去做内存的⾃动回收。
即在对象⽣成时维护⼀个对该对象引⽤次数的计数器,对象初次⽣成时计数器值为1,每增加⼀个到该对象的引⽤,计数器加1,每减少⼀个引⽤(如引⽤变量赋值null,或引⽤变量离开作⽤域),计数器减1,计数器为零时,对象内存会被⾃动回收。
该⽅法的问题是存在内存泄漏的隐患,如对象相互引⽤、循环引⽤等情况相互引⽤:public class ReferenceCountingGc {Object instance = null;public static void main(String[] args) {ReferenceCountingGc objA = new ReferenceCountingGc();ReferenceCountingGc objB = new ReferenceCountingGc();objA.instance = objB;objB.instance = objA;objA = null;objB = null;}} 例⼦中两个new出来的对象ReferenceCountingGc由于通过内部的变量instance引⽤着对⽅,两个对象的引⽤计数都为1。
jdk8gc机制

jdk8gc机制JDK 8的GC机制JDK 8(Java Development Kit 8)是Java平台的一个版本,其中包含了许多新的特性和改进。
其中一个重要的改进就是对垃圾回收(Garbage Collection,GC)机制的优化。
GC是Java内存管理的关键部分,它负责自动回收不再使用的对象,释放内存空间,以避免内存泄漏和程序崩溃。
在JDK 8中,GC机制经历了一些重大的改进和优化,以提高性能和减少内存占用。
下面将介绍一些JDK 8的GC机制的特点和优势。
1. G1垃圾回收器(Garbage First Garbage Collector):JDK 8引入了一种新的垃圾回收器,即G1垃圾回收器。
相比于之前的垃圾回收器,G1垃圾回收器具有更好的吞吐量和更低的延迟。
它能够自动监控各个内存区域的垃圾情况,并根据垃圾情况进行动态调整,以实现更高效的垃圾回收。
2. 基于Region的内存管理:JDK 8将堆内存分成了许多大小相等的区域(Region),每个区域独立管理。
这种基于Region的内存管理方式使得GC操作更加高效,可以更细粒度地管理内存,减少内存碎片。
同时,它还可以将GC操作分散到多个线程上,以提高并发性能。
3. 并行和并发回收:JDK 8的GC机制充分利用了多核处理器的并行能力。
它可以同时使用多个线程进行垃圾回收操作,以加快回收速度。
同时,它还引入了一些并发回收的策略,使得垃圾回收操作与应用程序的执行可以同时进行,减少了停顿时间,提高了用户体验。
4. 元空间代替永久代:JDK 8取消了永久代(PermGen)的概念,取而代之的是元空间(Metaspace)。
永久代是用于存储类的元数据信息的,而元空间则是使用本地内存(Native Memory)来存储类的元数据信息。
这种改变解决了永久代容易出现内存溢出的问题,提高了内存的利用率。
5. 垃圾回收日志和诊断工具:JDK 8提供了更丰富的垃圾回收日志和诊断工具,可以帮助开发人员更好地理解和分析程序的垃圾回收情况。
java对象回收方法

java对象回收方法在Java中,对象的回收主要依靠垃圾回收机制来完成。
垃圾回收是指自动释放不再被使用的内存空间,以便新的对象可以被创建和使用。
Java的垃圾回收机制通过跟踪对象的引用来确定哪些对象是活动的,哪些对象是可以被回收的。
Java对象的回收方法如下:1. 引用计数法(Reference Counting)引用计数法是一种简单的回收方法,在每个对象中维护一个引用计数器,记录对象被引用的次数。
当引用计数器为0时,表示该对象没有被引用,可以被回收。
但是,引用计数法无法解决循环引用的问题,例如两个对象相互引用,导致引用计数器无法为0,内存泄漏就会发生。
2. 可达性分析算法(Reachability Analysis)可达性分析是Java虚拟机使用的主要回收方法。
它通过从一组根对象(如线程栈、静态变量)出发,追踪对象之间的引用关系,标记可达的对象,然后将未标记的对象视为垃圾进行回收。
可达性分析考虑的是对象之间的引用关系,而不是引用计数,因此可以解决循环引用的问题。
3. 标记-清除算法(Mark-Sweep)标记-清除算法是可达性分析的一种实现方式。
首先,通过可达性分析标记出所有活动对象。
然后,将所有未标记的对象清除(回收)。
标记-清除算法会导致内存碎片的产生,在回收后,内存空间可能会被分割成多个不连续的小块,影响后续对象的分配。
标记-整理算法是可达性分析的另一种实现方式。
首先,通过可达性分析标记出所有活动对象。
然后,将所有活动对象向一端移动,将空间释放,并形成一块连续的内存空间。
标记-整理算法能够解决内存碎片的问题,但是会导致对象移动的开销。
5. 复制算法(Copying)复制算法是一种基于分代假设的垃圾回收算法。
它假设大部分对象的生命周期较短,并将堆内存划分为两个区域:From区和To区。
当From 区满时,将存活的对象复制到To区,并清除From区的所有对象。
复制算法具有简单高效的特点,但会浪费一半的内存空间。
java中stack和heap的区别,java中的垃圾回收机制

java中stack和heap的区别,java中的垃圾回收机制#. 在java中有两类内存。
分别称为stack(栈)和heap(堆)。
stack是程序内存空间,因此所有的基本类型和对象的引⽤是存在stack中。
heap是java虚拟机储存对象的,它是⼀个巨⼤的内存,当你创造⼀个对象,java虚拟机把对象放⼊heap中,把创造的对象的地址放⼊stack中。
因此,基本类型、对象的引⽤储存在stack中;对象储存在heap中。
#. java中的垃圾回收机制
当你new⼀个新的对象,java分配必需的内存。
当你⽤完⼀个对象时,java的垃圾回收器为你把内存收回。
垃圾回收以线程的形式在后台运⾏,寻找那些⽆有⽤引⽤(reference)的对象,发现之后便销毁对象,并收回内存。
垃圾回收是在java虚拟机间实现的,它们通常有相同的步骤,⾸先垃圾回收器获得正在运⾏的线程和所有已经加载的类的快照,
然后所有线程中涉及到的对象被标记为最近使⽤的,当可能涉及的对象都被标记的时候,剩下没标记的就被舍弃。
为了帮助虚拟机,我们主动移除⼀些不在需要的对象是⼀个不错的做法,可以通过将引⽤设置为null来实现。
eg:
Text t = new Test();
t.someAction();
//all done
t = null;。
JVM垃圾回收算法及G1回收机制

JVM垃圾回收算法及G1回收机制JVM(Java Virtual Machine)是Java程序运行的环境,其中的垃圾回收算法是JVM内存管理的重要组成部分。
垃圾回收算法的作用是自动释放不再使用的内存空间,以提高程序的性能和效率。
其中,G1(Garbage-First)是一种现代化的垃圾回收器,相较于传统的垃圾回收算法具有更高的效率和更低的延迟。
垃圾回收算法的核心思想是通过扫描内存,找出不再被引用的对象,并将其释放。
常见的垃圾回收算法包括标记-清除算法、复制算法、标记-整理算法等。
标记-清除算法是最基础的垃圾回收算法之一、它通过标记所有被引用的对象,然后清除未被标记的对象。
这个算法的优点是可以处理任意的内存分配情况,但是会产生大量的碎片化空间。
复制算法是另一种常见的垃圾回收算法。
它将内存分为两个区域,每次只使用其中一个区域。
当一个区域满了之后,将还存活的对象复制到另一个区域,然后清除当前区域。
这个算法的优点是简单高效,但是会浪费一半的内存空间。
标记-整理算法是标记-清除算法的改进版。
它先标记所有被引用的对象,然后将存活的对象向一端移动,然后清除边界之外的对象。
这个算法的优点是可以减少碎片化空间,但是会有对象移动的开销。
G1是一种基于标记-整理算法的垃圾回收器,它在Java SE 6u14版本中引入。
G1回收机制主要有以下几个特点:首先,G1将堆内存划分为若干个大小相等的区域(Region),每个区域可以是Eden区、Survivor区或Old区。
这种划分方式可以有效地减少碎片化问题,并且可以根据实际情况动态调整区域的大小。
其次,G1采用了增量式的标记算法,在应用程序运行的同时进行垃圾回收操作。
这样可以减少单次垃圾回收的暂停时间,并且将垃圾回收的工作均匀地分布在多个时间片段中,避免长时间的停顿。
再次,G1使用了全局的标记-整理算法。
它通过标记所有被引用的对象,然后将存活的对象向一端移动,并清除边界之外的对象。
Java8的GC垃圾回收

Java8的GC垃圾回收Java垃圾回收概况Java GC(Garbage Collection,垃圾回收)机制,是Java与C++/C的主要区别之⼀,作为Java开发者,⼀般不需要专门编写内存回收和垃圾清理代码,对内存泄露和溢出的问题,也不需要像C程序员那样战战兢兢。
这是因为在Java虚拟机中,存在⾃动内存管理和垃圾清扫机制。
概括地说,该机制对JVM中的内存进⾏标记,并确定哪些内存需要回收,根据⼀定的回收策略,⾃动的回收内存,永不停息的保证JVM 中的内存空间,防⽌出现内存泄露和溢出问题。
关于JVM,需要说明⼀下的是,⽬前使⽤最多的Sun公司的JDK中,⾃从1999年的JDK1.2开始直⾄现在仍在⼴泛使⽤的JDK6,其中默认的虚拟机都是HotSpot。
2009年,Oracle收购Sun,加上之前收购的EBA公司,Oracle拥有3⼤虚拟机中的两个:JRockit和HotSpot,Oracle也表明了想要整合两⼤虚拟机的意图,但是⽬前在新发布的JDK8中,默认的虚拟机仍然是HotSpot,因此本⽂中默认介绍的虚拟机都是HotSpot,相关机制也主要是指HotSpot的GC机制。
Java GC机制主要完成3件事:确定哪些内存需要回收确定什么时候需要执⾏GC如何执⾏GC经过这么长时间的发展,Java GC机制已经⽇臻完善,⼏乎可以⾃动的为我们做绝⼤多数的事情。
然⽽,如果我们从事较⼤型的应⽤软件开发,曾经出现过内存优化的需求,就必定要研究Java GC机制。
学习Java GC机制,可以帮助我们在⽇常⼯作中排查各种内存溢出或泄露问题,解决性能瓶颈,达到更⾼的并发量,写出更⾼效的程序。
我们将从4个⽅⾯学习Java GC机制,1,内存是如何分配的;2,如何保证内存不被错误回收(即:哪些内存需要回收);3,在什么情况下执⾏GC以及执⾏GC的⽅式;4,如何监控和优化GC机制。
内存是如何分配的这⾥所说的内存分配,主要指的是在堆上的分配,⼀般的,对象的内存分配都是在堆上进⾏,但现代技术也⽀持将对象拆成标量类型(标量类型即原⼦类型,表⽰单个值,可以是基本类型或String等),然后在栈上分配,在栈上分配的很少见,我们这⾥不考虑,接下来我们⼀起来了解下内存分区,对我们后⾯学习的有所帮助。
- 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
- 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
- 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。
Java垃圾回收机制1、引用计数:没有应用到JVM,只是理论说明。
2、活的对象:可以追溯到堆栈和静态存储区3、Stop and copy:停止工作,将活的对象从一个堆复制到另一个堆,紧密排列,然后剩下的就是垃圾对象了。
问题:(1)空间;(2)复制:垃圾少,要复制大量活的对象4、Mark and sweep:标记每个对象是否活对象,标记完所有对象,清楚非活的对象。
问题:空间不连续。
5、如果垃圾很少,复制效率降低的话就用,标记清扫;如果不连续空间太多的话,则转到停止复制6、优化方法:Just-In-Time 和HotSoft,前者翻译成本地机器码,后者惰性编译,只在必要时编译代码,不被执行的代码不被编译,代码每次执行的时候都会做一定的优化。
7、避免内存泄漏或者不当使用的方法:(1)重复使用已经初始化的对象,尽量少声明对象;(2)try catch finally语句中清楚不需要的对象;(3)执行完毕清楚不必要的对象,当然是那些引用可能被保存的对象。
尽管java虚拟机和垃圾回收机制管理着大部分的内存事务,但是在java软件中还是可能存在内存泄漏的情况。
的确,在大型工程中,内存泄漏是一个普遍问题。
避免内存泄漏的第一步,就是要了解他们发生的原因。
这篇文章就是要介绍一些常见的缺陷,然后提供一些非常好的实践例子来指导你写出没有内存泄漏的代码。
一旦你的程序存在内存泄漏,要查明代码中引起泄漏的原因是很困难的。
同时这篇文章也要介绍一个新的工具来查找内存泄漏,然后指明发生的根本原因。
这个工具容易上手,可以让你找到产品级系统中的内存泄漏。
垃圾回收(GC)的角色虽然垃圾回收关心着大部分的问题,包括内存管理,使得程序员的任务显得更加轻松,但是程序员还是可能犯些错误导致内存泄漏问题。
GC(垃圾回收)通过递归对所有从“根”对象(堆栈中的对象,静态数据成员,JNI句柄等等)继承下来的引用进行工作,然后标记所有可以访问的活动着的对象。
而这些对象变成了程序唯一能够操纵的对象,其他的对象都被释放了。
因为GC使得程序不能够访问那些被释放的对象,所以这样做是安全的。
内存管理可以说是自动的,但是这并没有让程序员脱离内存管理问题。
比方说,对于内存的分配(还有释放)总是存在一定的开销,尽管这些开销对程序员来说是隐含的。
一个程序如果创建了很多对象,那么它就要比完成相同任务而创建了较少对象的程序执行的速度慢(如果其他的条件都相同)。
文章更多想说的,导致内存泄漏主要的原因是,先前申请了内存空间而忘记了释放。
如果程序中存在对无用对象的引用,那么这些对象就会驻留内存,消耗内存,因为无法让垃圾回收器验证这些对象是否不再需要。
正如我们前面看到的,如果存在对象的引用,这个对象就被定义为―活动的‖,同时不会被释放。
要确定对象所占内存将被回收,程序员就要务必确认该对象不再会被使用。
典型的做法就是把对象数据成员设为null或者从集合中移除该对象。
注意,当局部变量不需要时,不需明显的设为null,因为一个方法执行完毕时,这些引用会自动被清理。
从更高一个层次看,这就是所有存在内存管的语言对内存泄漏所考虑的事情,剩余的对象引用将不再会被使用。
典型的泄漏既然我们知道了在java中确实会存在内存泄漏,那么就让我们看一些典型的泄漏,并找出他们发生的原因。
全局集合在大型应用程序中存在各种各样的全局数据仓库是很普遍的,比如一个JNDI-tree或者一个session table。
在这些情况下,注意力就被放在了管理数据仓库的大小上。
当然是有一些适当的机制可以将仓库中的无用数据移除。
可以有很多不同的解决形式,其中最常用的是一种周期运行的清除作业。
这个作业会验证仓库中的数据然后清除一切不需要的数据。
另一个办法是计算引用的数量。
集合负责跟踪集合中每个元素的引用者数量。
这要求引用者通知集合什么时候已经对元素处理完毕。
当引用者的数目为零时,就可以移除集合中的相关元素。
高速缓存高速缓存是一种用来快速查找已经执行过的操作结果的数据结构。
因此,如果一个操作执行很慢的话,你可以先把普通输入的数据放入高速缓存,然后过些时间再调用高速缓存中的数据。
高速缓存多少还有一点动态实现的意思,当数据操作完毕,又被送入高速缓存。
一个典型的算法如下所示:1.检查结果是否在高速缓存中,存在则返回结果;2.如果结果不在,那么计算结果;3.将结果放入高速缓存,以备将来的操作调用。
这个算法的问题(或者说潜在的内存泄漏)在最后一步。
如果操作是分别多次输入,那么存入高速缓存的内容将会非常大。
很明显这个方法不可取。
为了避免这种潜在的致命错误设计,程序就必须确定高速缓存在他所使用的内存中有一个上界。
因此,更好的算法是:1.检查结果是否在高速缓存中,存在则返回结果;2.如果结果不在,那么计算结果;3.如果高速缓存所占空间过大,移除缓存中旧的结果;4.将结果放入高速缓存,以备将来的操作调用。
通过不断的从缓存中移除旧的结果,我们可以假设,将来,最新输入的数据可能被重用的几率要远远大于旧的结果。
这通常是一个不错的设想。
这个新的算法会确保高速缓存的容量在预先确定的范围内。
精确的范围是很难计算的,因为缓存中的对象存在引用时将继续有效。
正确的划分高速缓存的大小是一个复杂的任务,你必须权衡可使用内存大小和数据快速存取之间的矛盾。
另一个解决这个问题的途径是使用ng.ref.SoftReference类来将对象放入高速缓存。
这个方法可以保证当虚拟机用完内存或者需要更多堆的时候,可以释放这些对象的引用。
类装载器Java类装载器创建就存在很多导致内存泄漏的漏洞。
由于类装载器的复杂结构,使得很难得到内存泄漏的透视图。
这些困难不仅仅是由于类装载器只与―普通的‖对象引用有关,同时也和对象内部的引用有关,比如数据变量,方法和各种类。
这意味着只要存在对数据变量,方法,各种类和对象的类装载器,那么类装载器将驻留在JVM中。
既然类装载器可以同很多的类关联,同时也可以和静态数据变量关联,那么相当多的内存就可能发生泄漏。
定位内存泄漏常常地,程序内存泄漏的最初迹象发生在出错之后,在你的程序中得到一个OutOfMemoryError。
这种典型的情况发生在产品环境中,而在那里,你希望内存泄漏尽可能的少,调试的可能性也达到最小。
也许你的测试环境和产品的系统环境不尽相同,导致泄露的只会在产品中暴露。
这种情况下,你需要一个低负荷的工具来监听和寻找内存泄漏。
同时,你还需要把这个工具同你的系统联系起来,而不需要重新启动他或者机械化你的代码。
也许更重要的是,当你做分析的时候,你需要能够同工具分离而使得系统不会受到干扰。
一个OutOfMemoryError常常是内存泄漏的一个标志,有可能应用程序的确用了太多的内存;这个时候,你既不能增加JVM的堆的数量,也不能改变你的程序而使得他减少内存使用。
但是,在大多数情况下,一个OutOfMemoryError是内存泄漏的标志。
一个解决办法就是继续监听GC的活动,看看随时间的流逝,内存使用量是否会增加,如果有,程序中一定存在内存泄漏。
详细输出有很多办法来监听垃圾回收器的活动。
也许运用最广泛的就是以:-Xverbose:gc选项运行JVM,然后观察输出结果一段时间。
[memory] 10.109-10.235: GC 65536K->16788K (65536K), 126.000 ms箭头后的值(在这个例子中16788K)是垃圾回收后堆的使用量。
控制台观察这些无尽的GC详细统计输出是一件非常单调乏味的事情。
好在有一些工具来代替我们做这些事情。
The JRockit Management Console可以用图形的方式输出堆的使用量。
通过观察图像,我们可以很方便的观察堆的使用量是否伴随时间增长。
Figure 1. The JRockit Management Console管理控制台甚至可以配置成在堆使用量出现问题(或者其他的事件发生)时向你发送邮件。
这个显然使得监控内存泄漏更加容易。
内存泄漏探测工具有很多专门的内存泄漏探测工具。
其中The JRockit Memory Leak Detector可以供来观察内存泄漏也可以针对性地找到泄漏的原因。
这个强大的工具被紧密地集成在JRockit JVM中,可以提供最低可能的内存事务也可以轻松的访问虚拟机的堆。
专门工具的优势一旦你知道程序中存在内存泄漏,你需要更专业的工具来查明为什么这里会有泄漏。
而JVM是不可能告诉你的。
现在有很多工具可以利用了。
这些工具本质上主要通过两种方法来得到JVM的存储系统信息的:JVMTI和字节码仪器。
Java虚拟机工具接口(JVMTI)和他的原有形式JVMPI(压型接口,profiling Interface)都是标准接口,作为外部工具同JVM 进行通信,搜集JVM的信息。
字节码仪器则是引用通过探针获得工具所需的字节信息的预处理技术。
通过这些技术来侦测内存泄漏存在两个缺点,而这使得他们在产品级环境中的运用不够理想。
首先,根据两者对内存的使用量和内存事务性能的降级是不可以忽略的。
从JVM获得的堆的使用量信息需要在工具中导出,收集和处理。
这意味着要分配内存。
按照JVM的性能导出信息是需要开销的,垃圾回收器在搜集信息的时候是运行的非常缓慢的。
另一个缺点就是,这些工具所需要的信息是关系到JVM的。
让工具在JVM开始运行的时候和它关联,而在分析的时候,分离工具而保持JVM运行,这显然是不可能的。
既然JRockit Memory Leak Detector是被集成到JVM中的,那么以上两种缺点就不再存在。
首先,大部分的处理和分析都是在JVM中完成的,所以就不再需要传送或重建任何数据。
处理也可以建立在垃圾回收器的基础上,即提高速度。
再有,内存泄漏侦测器可以同一个运行的JVM关联和分离,只要JVM在开始的时候伴随着–Xmanagement选项(通过远程JMX接口允许监听和管理JVM)。
当工具分离以后,工具不会遗留任何东西在JVM中;JVM就可以全速运行代码就好像工具关联之前一样。
趋势分析让我们更深一步来观察这个工具,了解他如何捕捉到内存泄漏。
在你了解到代码中存在内存泄漏,第一步就是尝试计算出什么数据在泄漏——哪个对象类导致泄露。
The JRockit Memory Leak Detector通过在垃圾回收的时候,计算每个类所包含的现有的对象来达到目的。
如果某一个类的对象成员数目随着时间增长(增长率),那么这里很可能存在泄漏。
Figure 2. The trend analysis view of the Memory Leak Detector因为一个泄漏很可能只是像水滴一样小,所以趋势分析必须运行足够长的一段时间。
在每个短暂的时间段里,局部类的增加会使得泄漏发生推迟。