JVM知识(四):GC配置参数
JVM配置参数分为三类参数:跟踪参数、堆分配参数、栈分配参数
这三类参数分别用于跟踪监控JVM状态,分配堆内存以及分配栈内存。
跟踪参数
跟踪参数用户跟踪监控JVM,往往被开发人员用于JVM调优以及故障排查。
1、当发生GC时,打印GC简要信息
使用-XX:+PrintGC或-verbose:gc参数
这两个配置参数效果是一样的,都是在发生GC时打印出简要的信息,例如:
public static void main(String[] args) {
byte[] bytes =null;
for(int i=0;i<100;i++){
bytes = new byte[1 * 1024 * 1024];
}
} 这个程序连续创建了100个1M的数组对象,使用-XX:+PrintGC或-verbose:gc参数执行该程序,即可查看到GC情况:
1: [GC (Allocation Failure) 32686K->1648K(123904K), 0.0007230 secs]
2: [GC (Allocation Failure) 34034K->1600K(123904K), 0.0009652 secs]
3: [GC (Allocation Failure) 33980K->1632K(123904K), 0.0005306 secs]
我们可以看到程序执行了3次GC(minor GC),这三次GC都是新生代的GC,(新生代GC的解释)因为这个程序每次创建新的数组对象,都会把新的对象赋给bytes变量,而老的对象没有任意对象引用它,老对象会变的不可达,这些不可达的老对象在新生代minor GC时候被回收掉。总结新生代GC:新生代GC回收一些不可达(没有任意对象引用)的老对象
32686K表示回收前,对象占用空间。1648K表示回收后,对象占用空间。123904K表示还有多少空间可用。0.0007230 secs表示这次垃圾回收花的时间。
2、打印GC的详细信息以及堆使用详细信息
使用-XX:+PrintGCDetails参数
1: [GC (Allocation Failure) [PSYoungGen: 32686K->1656K(37888K)] 32686K->1664K(123904K), 0.0342788 secs] [Times: user=0.00 sys=0.00, real=0.03 secs]
2: [GC (Allocation Failure) [PSYoungGen: 34042K->1624K(70656K)] 34050K->1632K(156672K), 0.0013466 secs] [Times: user=0.00 sys=0.00, real=0.00 secs]
3: Heap (以下是堆使用详细信息) (新生代、老生代、元空间)
4: PSYoungGen total 70656K, used 43118K [0x00000000d6100000, 0x00000000dab00000, 0x0000000100000000) (新生代:伊甸区(eden)和幸存区(from和to))
5: eden space 65536K, 63% used [0x00000000d6100000,0x00000000d8985ac8,0x00000000da100000) 伊甸区(eden)
6: from space 5120K, 31% used [0x00000000da600000,0x00000000da796020,0x00000000dab00000) 幸存区(from和to)
7: to space 5120K, 0% used [0x00000000da100000,0x00000000da100000,0x00000000da600000) 幸存区(from和to)
8: ParOldGen total 86016K, used 8K [0x0000000082200000, 0x0000000087600000, 0x00000000d6100000)(老生代)
9: object space 86016K, 0% used [0x0000000082200000,0x0000000082202000,0x0000000087600000)
10: Metaspace used 2669K, capacity 4486K, committed 4864K, reserved 1056768K (元空间)
11: class space used 288K, capacity 386K, committed 512K, reserved 1048576K
我们看到除了打印GC信息之外,还显示了堆使用情况,堆分为新生代、老年代、元空间。注意这里没有永久区了,永久区在java8已经移除,原来放在永久区的常量、字符串静态变量都移到了元空间,并使用本地内存。
新生代当中又分为伊甸区(eden)和幸存区(from和to),从上面打印的内容可以看到新生代总大小为70656K,使用了43118K,细心的同学的可能会发现eden+from+to=65536K+5120K+5120K=75776 并不等于总大小70656K,这是为什么呢?这是因为新生代的垃圾回收算法是采用复制算法,简单的说就是在from和to之间来回复制(复制过程中再把不可达的对象回收掉),所以必须保证其中一个区是空的,这样才能有预留空间存放复制过来的数据,所以新生代的总大小其实等于eden+from(或to)=65536K+5120K=70656k。
3、使用外部文件记录GC的日志
还有一个非常有用的参数,它可以把GC的日志记录到外部文件中,这在生产环境进行故障排查时尤为重要,当java程序出现OOM时,总希望看到当时垃圾回收的情况,通过这个参数就可以把GC的日志记录下来,便于排查问题,当然也可以做日常JVM监控。
-Xloggc:log/gc.log
4、监控类的加载
-XX:+TraceClassLoading
使用这个参数可以监控java程序加载的类:
堆配置参数
指定最大堆,最小堆:Xmx、Xms
这两个参数是我们最熟悉最常用的参数,可以用以下代码打印出目前内存使用的情况:
public static void main(String[] args) {
System.out.println("最大堆:"+Runtime.getRuntime().maxMemory()/1024/1024+"M");
System.out.println("空闲堆:"+Runtime.getRuntime().freeMemory()/1024/1024+"M");
System.out.println("总的堆:"+Runtime.getRuntime().totalMemory()/1024/1024+"M");
}
最大堆也就是Xmx参数指定的大小,表示java程序最大能使用多少内存大小,如果超过这个大小,那么java程序会报:out of memory(OOM错误),空闲堆表示程序已经分配的内存大小减去已经使用的内存大小,而总的堆表示目前程序已经配置到多少内存大小,一般而言程序一启动,会按照-Xms5m先分配5M的空间,这时总的堆大小就是5M。
指定新生代内存大小:Xmn,例如我们指定-Xmx20m -Xms5m -Xmn2m -XX:+PrintGCDetails
可以看到新生代总大小为eden+from+to=1024k+512k+512k=2M,和我们设置的-Xmn相对应。
新生代(eden+from+to)和老年代(不包含永久区)的比值:-XX:NewRatio
例如我们设置参数:-Xmx20m -Xms20m -XX:NewRatio=4 -XX:+PrintGCDetails(注意这里改参数为4表示新生代和老年代比值为1:4)
可以看到新生代:eden+from+to=3072+512+512=4096k,老年代:16384k。新生代:老年代=4096k:16384k =1:4 和-XX:NewRatio=4吻合。
Eden区与Survivor区(from、to)的大小比值:-XX:SurvivorRatio(如设置为8,则两个Survivor区与一个Eden区的比值为2:8,一个Survivor区占整个年轻代的1/10)
例如设置参数-Xmx20m -Xms20m -Xmn8m -XX:SurvivorRatio=6 -XX:+PrintGCDetails
这个参数设置了新生代内存大小为8m,并设置Survivor区与一个Eden区的比值为2:6,来看看打印信息:
Survivor区=from+to=2048,Eden区=6144K,Survivor区:Eden区=2:6,和-XX:SurvivorRatio=6吻合。
其他还有-XX:+HeapDumpOnOutOfMemoryError、-XX:+HeapDumpPath这两个参数可以实现在发生OOM异常时把堆栈信息打印到外部文件。
堆分配参数的总结
根据实际事情调整新生代和幸存代的大小
官方推荐新生代占堆的3/8
幸存代占新生代的1/10
在OOM时,记得Dump出堆,确保可以排查现场问题
永久区分配参数
-XX:PermSize -XX:MaxPermSize
用于设置永久区的初始空间和最大空间,他们表示一个系统可以容纳多少个类型,一般空间比较小。在java1.8以后,永久区被移到了元数据区,使用本地内存,所以这两个参数也不建议再使用。
栈大小分配参数
栈大小参数为-Xss,通常只有几百k,决定了函数调用的深度,每个线程都有自己独立的栈空间。如果函数调用太深,超过了栈的大小,则会抛出java.lang.StackOverflowError,通常我们遇到这种错误,不是去调整-Xss参数,而是应该去调查函数调用太深的原理,是否使用递归,能不能保证递归出口等。
JVM知识(四):GC配置参数的更多相关文章
- java jvm内存管理/gc策略/参数设置
1. JVM内存管理:深入垃圾收集器与内存分配策略 http://www.iteye.com/topic/802638 Java与C++之间有一堵由内存动态分配和垃圾收集技术所围成的高墙,墙外面的人想 ...
- 深入理解JVM(三)——配置参数
JVM配置参数分为三类参数: 1.跟踪参数 2.堆分配参数 3.栈分配参数 这三类参数分别用于跟踪监控JVM状态,分配堆内存以及分配栈内存. 跟踪参数 跟踪参数用于跟踪监控JVM,往往被开发人员用于J ...
- JVM内存模型及配置参数
JVM 分为堆.栈.方法区.程序计数器.本地方法栈 栈内存存放局部变量表.操作栈.动态链接.方法出口等信息 1. 局部变量表存放了编译期可知的各种基本数据类型(boolean.byte.char.s ...
- java面试-JVM常用的基本配置参数有哪些?
1.-Xms 初始大小内存,默认为物理内存 1/64,等价于 -XX:InitialHeapSize 2.-Xmx 最大分配内存,默认为物理内存的 1/4,等价于 -XX:MaxHeapSize 3. ...
- Elasticsearch笔记四之配置参数与核心概念
在es根目录下有一个config目录,在此目录下有两个文件分别是elasticsearch.yml和logging.yml. logging.yml是日志文件,es也是使用log4j来记录日志的,我在 ...
- JVM虚拟机(1)---常用JVM配置参数
常用JVM配置参数 常用JVM配置参数主要有:Trace跟踪参数.堆的分配参数.栈的分配参数. 一.Trace跟踪参数 跟踪参数用于跟踪监控JVM,对于开发人员来讲用于JVM调优以及故障排查的. 1. ...
- JVM配置参数详解
记录一下jvm中的一些配置参数,这些肯定不全的,希望你们能留个言,补全一下,谢谢啦! -XX: MaxDirectMemorySize--->设置直接内存,不设置与Java堆内存最大值一致 -X ...
- JVM学习八:常用JVM配置参数
前面学习的都是和类加载相关的知识,接下来学习的则和GC相关的知识,都是JVM的几个重点块. 零.在IDE的后台打印GC日志: 既然学习JVM,阅读GC日志是处理Java虚拟机内存问题的基础技能,它只是 ...
- CMS GC启动参数优化配置
简介: java启动参数共分为三类: 其一是标准参数(-),所有的JVM实现都必须实现这些参数的功能,而且向后兼容: 其二是非标准参数(-X),默认jvm实现这些参数的功能,但是并不保证所有jvm实现 ...
随机推荐
- Vue使用过渡类名实现动画和自定义前缀
Vue使用过渡类名实现动画和自定义前缀 1.效果演示 2.相关代码 <!DOCTYPE html> <html lang="en"> <head> ...
- Android Layout 01_activity_Login.xml
activity_login.xml <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android ...
- java基本类型自动转换
具体自动提升类型如上图所示.其中long->float的转换一开始让我感觉有点问题,因为long是64位的,而float却是32位的. 遂找寻答案,参考博客java中long到float的自动转 ...
- C语言中函数返回字符串的四种方法
在讨论着四种方法之前,首先要对函数有一个简单的认识,无论是在形实结合时,还是在return语句返回时,都有一个拷贝的过程.你传进来的参数是个值,自然函数在工作之前要把这个值拷贝一份供自己使用,你传进来 ...
- Oracle数据库 中的基础的一些语法结构
方括号里的内容为可选项 大括号是必填 1PL/SQL结构块 DECLARE /* * 声明部分——定义常量.变量.复杂数据类型.游标.用户自定义异常 */ BEGIN /* * 执行部分——PL/SQ ...
- 美化checkbox多选框
看到那些UI框架都是有美化checkbox多选框的,不过大多是图片或者是字体图标.于是就利用label仿了个多选框效果. <!DOCTYPE html> <html lang=&qu ...
- [Linux]——进程管理相关
一些概念 程序program:通常以二进制程序放置在存储媒介中,以物理文件形式存在 进程process:程序通过用户执行被触发后,执行者的权限与属性.程序的代码和所需数据会被加载到内存中,OS给予这个 ...
- Linux man C++ 库函数
默认情况下,linux是的man是不能查阅C++的标准库函数的,这个很不方便,那有没有办法可以直接man C++标准库函数呢? 当然有,不过要自己动手,自己动手,才能丰衣足食! 1. 下载安装manp ...
- spark集群搭建(java)未完待续
环境 操作系统:windows10 虚拟机工具:VMware14.1 NUX版本:Centos7.2(64) JDK:1.8(64) 一.安装linux,master(桥接模式上网),slave(na ...
- c#基础学习(0703)之string.Format格式化日期
C# string.Format格式化日期 DateTime dt = ,,,,,,); string.Format("{0:y yy yyy yyyy}",dt); //17 1 ...