按月归档:2012年五月


JVM之垃圾回收机制

一. 收集器 JVM通过GC来回收堆和方法区中的内存,通常采用收集器的方式实现GC,主要的收集器有引用计数收集器和跟踪收集器。 1、 引用计数收集器 引用计数收集器采用分散式的管理方式,通过计数器记录对象是否被引用。 引用计数收集器需要在每次对象赋值时进行引用计数器的增减,有一定消耗;此外,引用计数收集器对于循环引用的场景无法实现回收。因此,Sun JDK在实现GC时未采用此种方式。 2、 跟踪计数器 跟踪计数器采用集中式的管理方式,全局记录数据的引用状态。主要有复制(Copying)、标记-清除(Mark-Sweep)和标记-压缩(Mark-Compact)三种实现算法:     1) 复制(Copying)     复制从根集合中扫描出存活的对象,并将找到的存活对象复制到一块新的未完全使用的空间中。    2) 标记-清除(Mark-Sweep)     标记-清除从根集合开始扫描,对存活的对象进行标记,扫描完毕后,再扫描整个空间中未标记的对象,并进行回收。     3) 标记-压缩(Mark-Compact)     标记-压缩采用和标记-清除一样的方式对存活对象进行标记,但在清除时有所不同。在回收不存活对象所占用的内存空间后,会将其他所有存活对象都往左端空闲的空间进行移动,并更新引用其对象的指针。 二、Sun JDK中可用的GC Sun JDK中可用的GC方式如下图所示:

java 内存区域

java虚拟机栈       虚拟机栈描述的是java方法执行的内存模型:每个方法被执行的时候都会同时创建一个栈帧(Stack Frame)用于存储局部变量表、操作数栈、动态链接、方法出口等信息。每个方法被调用直至执行完成的过程,就对应着一个栈帧的虚拟机中从入栈的过程。 java堆       java堆(java heap)是java虚拟机所管理的内存中最大的一块。java堆是被所有线程共享的一块内存区域,在虚拟机启动时创建。此内存区域的唯一目的就是存放对象实例,几乎所有的对象实例都存这里分配内存。这一点在java虚拟机规范的中的描述是:所有的对象实例以级数组都要在堆上分配,但是随JIT编译器的发展与逃逸分析技术的逐渐成熟,栈上分配、标量替换优化技术将会导致一些微妙的变化发生,所有对象都分配在堆上也渐渐变得不是那么“绝对”了。       java堆中还可以细分为:新生代和老年代;再细致一点的有Eden空间、From Survivor空间和To Survivor空间。