JVM垃圾回收(GC)原理
一.基本垃圾回收算法
1.引用计数(Reference Counting)
比较古老的回收算法。原理是此对象有一个引用则增加一个引用计数,删除一个引用则较少一个引用计数。垃圾回收时,只回收引用计数为0的对象。此算法最致命的是无法处理互相引用的问题。
2.标记-清除(Mark-Sweep)
此算法执行分两个阶段。第一个阶段是从根节点开始标记所有被引用的对象;第二个阶段遍历整个堆,把未标记的对象清除。此算法需要暂停整个应用,并且会产生内存碎片。
3.复制(Copying)
此算法把内存空间划分为2个相同的区域,每次只使用其中一个区域。垃圾回收时,区域A从根开始访问每一个关联的正在使用的对象,并且把正在使用的对象复制到另外一个区域B,然后回收区域A中所有对象。此算法只访问活跃对象,所以遍历空间成本低,但是此算法的缺点也很明显,就是需要2倍内存空间。
4.标记-整理(Mark-Compact)
此算法也是分两个阶段。第一个阶段是从根节点开始标记所有被引用的对象;第二阶段遍历整个堆,清除未被标记的对象并且把标记的对象(正在使用的对象)压缩到一起按顺序存放。此算法避免了“标记-清除”算法中的内存碎片问题,并且也避免了“复制”算法的空间问题。
二.JVM内存模型。

1.基础概念
(1)JVM内存限制(最大值):
JVM内存的最大值跟操作系统有很大的关系。简单的说32位操作系统虽然可控内存有4GB,但是操作系统会给一个限制,Windows下为2GB,Linux下为3GB,而64位及以上的操作系统没有限制。
(2)JVM默认内存分配:
JVM初始分配的内存由-Xms指定:默认是物理内存的1/64但是小于1GB。
JVM最大分配的内存有-Xmx指定:默认是物理内存的1/4但是小于1GB。
(3)JVM内存自动调整:
默认空余堆内存小于40%时,JVM就会增大堆直到-Xmx限制,可以由-XX:MinHeapFreeRatio=指定。
默认空余堆内存大于70%时,JVM就是减少堆直到-Xms限制,可以由-XX:MaxHeapFreeRatio=指定。
但是服务器一般都设置-Xms、-Xmx相等以避免每次gc后都调整堆的大小,所以这两个参数一般没什么作用。
2.堆(Heap)和非堆(Non-heap)内存
(1)堆(Heap)

(2)非堆(Non-heap)

参考文档:
http://www.cnblogs.com/redcreen/tag/jvm/
http://chenchendefeng.iteye.com/blog/455883
http://www.open-open.com/lib/view/open1324736648468.html
http://vanadiumlin.iteye.com/blog/1267857
JVM参数:
1、JVM堆相关参数
| 参数 | 说明 | 备注 |
| -Xmx |
设置最大堆内存 |
一般-Xmx与-Xms设置相同 |
| -Xms |
设置最小堆内存 |
|
| -Xmn |
设置新生代内存大小 |
-Xmn等同于设置了相同的-XX:NewSize与-XX:MaxNewSize |
| -XX:NewSize |
设置新生代初始大小 |
|
| -XX:MaxNewSize | 设置新生代最大值 | |
| -XX:PermSize | 设置持久代初始大小 | 用于JDK8 之前 |
| -XX:MaxPermSize | 设置持久代的最大值 | 用于JDK8 之前 |
| -XX:MetaspaceSize | 设置持久代初始大小 | 用于JDK8之后,包含JDK8 |
| -XX:MaxMetaspaceSize | 设置持久代的最大值 | 用于JDK8之后,包含JDK8 |
| -Xss | 设置线程栈的大小 | 线程栈所占内存不是堆内存,是操作系统内存减去堆内存 |
| -XX:MinHeapFreeRatio | 设置堆空间最小空闲比例 | 当堆空间的空闲内存小于这个数值时,JVM便会扩展堆空间,默认值40 |
| -XX:MaxHeapFreeRatio | 设置堆空间最大空闲比例 | 当堆空间的空闲内存大于这个数值时,JVM便会压缩堆空间,得到一个较小的堆,默认值70 |
| -XX:NewRatio | 设置老年代与新生代比例 | 等于老年代大小除以新生代大小 |
| -XX:SurviorRatio | 新生代中Eden与survivor区的比例 | |
| -XX:TargetSurvivorRatio | 设置survivor区的使用率 | 当survivor区的空间使用率达到这个数值时,会将对象送入老年代,默认值50,最高90 |
2、JVM垃圾收集器相关参数



| 参数 | 说明 | 备注 |
| -XX:+CMSClassUnLoadingEnabled | 允许对类元数据进行回收 | |
| -XX:CMSInitiatingPermOccupancyFraction | 当永久代占用率达到这一百分比时,启动CMS回收 | 前提是-XX:+CMSClassUnLoadingEnabled激活了 |
| -XX:UseCMSInitiatingOccupancyOnly | 表示只在达到阈值的时候,才进行CMS回收 | |
| -XX:+CMSIncrementalMode | 使用增量模式,比较适合单CPU |
3、实用JVM参数
| 参数 | 说明 | 备注 |
| -XX:+HeapDumpOnOutOfMemoryError | 当程序发生OOM时,导出应用程序当前堆快照 | |
| -XX:HeapDumpPath | 可以指定堆快照的保存位置 |
-XX:+HeapDumpOnOutOfMemoryError |
| -XX:OnOutOfMemoryError | 系统发生OOM错误时可运行一段第三方脚本 |
-XX:OnOutOfMemoryError=c:\reset.bat |
| -verbose:gc | 获取gc信息 | |
| -XX:+PrintGC | 获取gc信息 | |
| -XX:+PrintGCDetails | 获取gc详细信息 | |
| -XX:+PrintHeapAtGC | gc时打印详细的堆信息 | |
| -Xloggc | 将gc信息保存到文件 | -Xloggc:/Users/yangyu/Downloads/gc.log |
| -XX:+TraceClassLoading | 用于跟踪类加载情况 | |
| -XX:+TraceClassUnLoading | 用于跟踪类卸载情况 | |
| -verbose:class | 用于跟踪类加载和卸载情况 | |
| -XX:+DisableExplicitGC | 用于禁止显示的gc操作 | 也就是禁止代码里面的System.gc() |
JVM垃圾回收(GC)原理的更多相关文章
- 性能测试三十五:jvm垃圾回收-GC
垃圾回收-GC 三个问题 哪些内存需要回收? 什么时候回收? 如何回收? YoungGC和FullGC: 新生代引发的GC叫YoungGC 老年代引发的GC叫FullGC FullGC会引起整个Jvm ...
- .NET垃圾回收(GC)原理
作为.NET进阶内容的一部分,垃圾回收器(简称GC)是必须了解的内容.本着“通俗易懂”的原则,本文将解释CLR中垃圾回收器的工作原理. 基础知识 托管堆(Managed Heap) 先来看MSDN的解 ...
- JVM 垃圾回收GC Roots Tracing
1.跟搜索算法: JVM中对内存进行回收时,需要判断对象是否仍在使用中,可以通过GC Roots Tracing辨别. 定义: 通过一系列名为”GCRoots”的对象作为起始点,从这个节点向下搜索,搜 ...
- JVM垃圾回收(GC)流程
/* 首先介绍一下JVM中堆内存的组成: JVM堆内存主要由三部分组成: (1)新生代: 伊甸园区,存活区,伸缩区 (2)老年代: 老年区,伸缩区 (3)元空间(永久代): 元空间,伸缩区 注意:JD ...
- JVM垃圾回收——GC
一.JVM内存分配与回收 下图为堆内存结构图(注意:元数据区(MetaData )实际上不属于堆): 1.对象优先在Eden区分配 大多数情况下,对象在新生代中Eden区分配.当Eden区没有足够空间 ...
- JVM垃圾回收GC
1.堆的分代和区域 (年轻代)Young Generation(eden.s0.s1 space) Minor GC (老年代)Old Generation (Tenured space) ...
- JVM—垃圾回收GC算法
1 GC算法简介 算法 特点 标记-清除 分为"标记"和"清除"两个阶段 复制 可以解决效率问题,将可用的内存按容量划分为大小相等的两块. 标记-整理 先标记. ...
- JVM 及 垃圾回收机制原理
JVM Java 虚拟机 Java 虚拟机(Java virtual machine,JVM)是运行 Java 程序必不可少的机制.JVM实现了Java语言最重要的特征:即平台无关性.原理:编译后的 ...
- jvm垃圾回收原理(转)
原文链接:jvm垃圾回收原理 在jvm中堆空间划分为三个代:年轻代(Young Generation).年老代(Old Generation)和永久代(Permanent Generation).年轻 ...
- Java:JVM垃圾回收(GC)机制
JVM垃圾回收算法 1.标记清除(Mark-Sweep) 原理: 从根集合节点进行扫描,标记出所有的存活对象,最后扫描整个内存空间并清除没有标记的对象(即死亡对象)适用场合: 存活对象较多的情况下比较 ...
随机推荐
- JS获取剪贴板图片之后的格式选择与压缩问题
前言 某年某月的某一天,突然发现博客服务器上上传的图片都比较大,一些很小的截图都有几百kb,本来服务器带宽就慢,不优化一下说不过去. 问题细述 特别说明:本文代码因为只是用于我自己后台写markdow ...
- Sharing A Powerful Tool For Calculate Code Lines
最近正好需要统计下某项目代码行数,然后就找代码行数统计工具.以前找到过一个正则表达式,但是只有在VS2010下有用,VS2012和VS2013下的统计就不好使了. 接着搜索了一下代码行数统计绿色工具免 ...
- windows命令——explorer
转至http://www.cnblogs.com/ymind/archive/2012/03/30/explorer-command-args.html 今天才知道,explorer原来可以这样用, ...
- Linux study
在centos5.5中编译LNMP环境 一.配置好ip, dns, 网关, 确保使用远程连接工具能够连接服务器 centos设置ip地址,网关, dns教程: http://www.osyumwei. ...
- c#设计模式-适配器模式
一. 适配器(Adapter)模式 适配器模式把一个类的接口变换成客户端所期待的另一种接口,从而使原本接口不匹配而无法在一起工作的两个类能够在一起工作. 名称由来 这很像变压器(Adapter),变压 ...
- 【hadoop摸索系列】记录使用libhdfs访问hdfs的关键问题
hadoop官方的二进制发布版本一直是32位平台编译的,对于java来说跨平台不影响使用,但是为了在c/c++程序中操作hdfs就做不到了,因为libhdfs.so是二进制不兼容的. 我使用的是sta ...
- Sql Server系列:字符串函数
字符串函数用于对字符和二进制字符串进行各种操作,大多数字符串函数只能作用于char.nchar.varchar和nvarchar数据类型.字符串函数可以用在SELECT或者WHERE语句中. 1. A ...
- php的mysql\mysqli\PDO(二)mysqli
原文链接:http://www.orlion.ga/1147/ mysqli有面向对象风格和面向过程风格,个人感觉还是用面向对象风格比较好(毕竟是面向对象) 1.mysqli::_construct( ...
- 前端学PHP之字符串函数
× 目录 [1]特点 [2]输出 [3]空格[4]大小写[5]HTML[6]格式化[7]比较 前面的话 字符串的处理和分析在任何编程语言中都是一个重要的基础,往往是简单而重要的.信息的分类.解析.存储 ...
- ImageView设置边框的两种方式
转载:http://www.2cto.com/kf/201308/239945.html package cc.testimageviewbounds; import android.os.Bundl ...