Java虚拟机系列(四)---查看GC日志
这一节穿插一点如何在eclipse中配置并查看某个Java应用GC日志的知识点,我也是通过调研知道的,因为书中写的不是很详细,主要是为下一节做准备。
一、eclipse中配置GC
在eclipse中如果要给某个应用配置GC日志,首先右击该应用(main方法所在的类)->Run As->Run Configurations->Arguments,在VM arguments中配置如下参数:
-Xms20m //最小堆内存
-Xmx20m //最大堆内存
-XX:+PrintGCTimeStamps //打印GC时间
-XX:+PrintGCDetails //打印GC的详细信息
-verbose:gc //开启GC日志
-Xloggc:d:/gc4.log //GC日志存放的位置,后面就是查看这个日志文件
-Xmn10M //新生代内存区域的位置
-XX:SurvivorRatio=8 //在复制算法中Eden区和Survivor区的比例(后续章节会细说)
如下图所示:

配置完成之后点击Applay->Run,等程序运行完之后可以到刚才配置的日志路径中,找到相应的文件进行查看。
二、查看GC
下面以一个具体的实例来查看并分析一下GC日志
1、创建三个互相引用的对象实例

2、配置GC

3、查看日志内容
Java HotSpot(TM) 64-Bit Server VM (25.121-b13) for windows-amd64 JRE (1.8.0_121-b13), built on Dec 12 2016 18:21:36 by "java_re" with MS VC++ 10.0 (VS2010)
Memory: 4k page, physical 4057600k(480696k free), swap 8113324k(3128576k free)
CommandLine flags: -XX:InitialHeapSize=20971520 -XX:MaxHeapSize=20971520 -XX:MaxNewSize=10485760 -XX:NewSize=10485760 -XX:+PrintGC -XX:+PrintGCDetails -XX:+PrintGCTimeStamps -XX:SurvivorRatio=8 -XX:+UseCompressedClassPointers -XX:+UseCompressedOops -XX:-UseLargePagesIndividualAllocation -XX:+UseParallelGC
0.121: [GC (System.gc()) [PSYoungGen: 4219K->680K(9216K)] 4219K->3760K(19456K), 0.0024028 secs] [Times: user=0.00 sys=0.00, real=0.00 secs]
0.124: [Full GC (System.gc()) [PSYoungGen: 680K->0K(9216K)] [ParOldGen: 3080K->3639K(10240K)] 3760K->3639K(19456K), [Metaspace: 2587K->2587K(1056768K)], 0.0062622 secs] [Times: user=0.06 sys=0.00, real=0.01 secs]
Heap
PSYoungGen total 9216K, used 82K [0x00000000ff600000, 0x0000000100000000, 0x0000000100000000)
eden space 8192K, 1% used [0x00000000ff600000,0x00000000ff614920,0x00000000ffe00000)
from space 1024K, 0% used [0x00000000ffe00000,0x00000000ffe00000,0x00000000fff00000)
to space 1024K, 0% used [0x00000000fff00000,0x00000000fff00000,0x0000000100000000)
ParOldGen total 10240K, used 3639K [0x00000000fec00000, 0x00000000ff600000, 0x00000000ff600000)
object space 10240K, 35% used [0x00000000fec00000,0x00000000fef8de58,0x00000000ff600000)
Metaspace used 2593K, capacity 4486K, committed 4864K, reserved 1056768K
class space used 286K, capacity 386K, committed 512K, reserved 1048576K
上面的就是生成的GC日志,其中:
前两行是虚拟机的一些信息,如使用的是HotSpot虚拟机,JRE版本等;第三行是刚才配置的参数信息,真正的GC信息是第4、5行,所以这里直接将4、5两行单独拿出来进行分析:
0.121: [GC (System.gc()) [PSYoungGen: 4219K->680K(9216K)] 4219K->3760K(19456K), 0.0024028 secs] [Times: user=0.00 sys=0.00, real=0.00 secs]
0.124: [Full GC (System.gc()) [PSYoungGen: 680K->0K(9216K)] [ParOldGen: 3080K->3639K(10240K)] 3760K->3639K(19456K), [Metaspace: 2587K->2587K(1056768K)], 0.0062622 secs] [Times: user=0.06 sys=0.00, real=0.01 secs]
1)最前面的数字0.121、0.124代表了GC发生的时间,这个数字的含义是从JAVA虚拟机启动以来经过的秒数;
2)GC日志开头的“[GC”和“[Full GC”说明了这次垃圾收集的停顿类型,而不是用来区分新生代GC还是老年代GC。如果有"Full",说明这次GC是发生了Stop-The-World的,就是指在执行GC操作的时候,会停止掉当前正在进行的其他线程,有一种“这是我的地盘,你们都给我让开”的霸道有木有?如果是调用System.gc()的方法所触发的收集,将会显示“Full GC(System.gc)",本文的例子就是使用System.gc()进行的收集;
3)接下来的"[PSYoungGen"、“[ParOldGen”、“[Metaspace“标示GC发生的区域,这里显示的区域名称与所使用的GC收集器密切相关(以后会对垃圾收集器进行详细介绍),下面是常见收集器所代表的的区域名称:
1>如果使用的是Serial(串行)收集器,新生代名为Default New Generation ,显示就是DefNew
2>如果使用的是ParNew(并行)收集器,新生代名称就会变为[Parnew,意思为Parallel New Generation
3>如果采用的是Parallel Scavenge收集器,新生代名称为PSYongGen
老年代和永久代同理,名称由垃圾收集器决定
如上面的日志中新生代名称为[PSYoungGen,使用的是Parallel Scavenge收集器;老年代名称为[ParOldGen,使用的是Paralled Old收集器(Parallel Scavenge收集器的老年代版本),还有元空间区域,有些文章中使用的是”[PermGen“,这时jdk8之前的版本,由于我使用的是jdk8,所以已经被”[Metaspace“所代替,至于为什么被代替,可以参考篇文章:https://blog.csdn.net/zhushuai1221/article/details/52122880
4)每个区域冒号后面,中括号内部的”680K->0K(9216K)]“含义是”GC前该内存区域已使用容量->GC后该内存区域已使用容量(该内存区域总容量)“。而在方括号之外的”3760K->3639K(19456K)“表示的是"GC前Java堆已使用容量->GC后Java堆已使用容量(Java堆总容量)"。
5)再往后 0.0062622 secs表示该内存区域GC所占用的时间单位是秒,有的收集器会给出具体的时间,如本例中的”[Times: user=0.06 sys=0.00, real=0.01 secs]“,这里面的user、sys、real与Linux命令所输出的时间含义一直,分别代表用户态消耗的CPU时间、内核态消耗的CPU时间和操作从开始到结束所经过的墙钟时间(Wall Clock Time)
TIPS:墙钟时间与CPU时间的区别是,墙钟时间包括各种非运算的等待耗时,例如等待磁盘的IO、等待线程阻塞;而CPU时间不包括这些耗时,但当系统中有对CPU或者多核的话,多线程操作会叠加这些CPU时间,所以读者看到user或sys的时间超过real时间是完全正常的。
上面对GC日志中各从参数的介绍摘自《深入理解Java虚拟机(第二版)》
再往下看,从第6行开始到结束的内容是各内存区域容量的变化情况!
备注:以上内容来源于《深入理解Java虚拟机》,根据自己的理解来总结,有些内容和书本内容相同是因为只有这么描述才比较好懂,毕竟大神还是大神,是颜色不一样的烟火!!
Java虚拟机系列(四)---查看GC日志的更多相关文章
- Java虚拟机系列——检视阅读
Java虚拟机系列--检视阅读 参考 java虚拟机系列 入门掌握JVM所有知识点 2020重新出发,JAVA高级,JVM JVM基础系列 从 0 开始带你成为JVM实战高手 Java虚拟机-垃圾收集 ...
- 【java虚拟机系列】java虚拟机系列之JVM总述
我们知道java之所以能够快速崛起一个重要的原因就是其跨平台性,而跨平台就是通过java虚拟机来完成的,java虚拟机属于java底层的知识范畴,即使你不了解也不会影响绝大部分人从事的java应用层的 ...
- 罗汉果与Java虚拟机系列目录与说明
声 明 罗汉果与Java虚拟机系列博文仅为本银结构性整合Java虚拟机知识的笔记和日常JVM问题的DEBUG记录.放到网上主要是为了方便自己今后查看.顺带能帮助到别人就更奈斯了. 目 录 ...
- 自制Java虚拟机(四)-对象、new、invokespecial
自制Java虚拟机(四)-对象.new.invokespecial 一.对象的表示 刚开始学Java的时候,图书馆各种教程,书名往往都是“Java面向对象高级编程”,通常作者都会与C++做个比较,列出 ...
- Java虚拟机系列一:一文搞懂 JVM 架构和运行时数据区
前言 之前写博客一直比较随性,主题也很随意,就是想到什么写什么,对什么感兴趣就写什么.虽然写起来无拘无束,自在随意,但也带来了一些问题,每次写完一篇后就要去纠结下一篇到底写什么,看来选择太多也不是好事 ...
- 【java虚拟机系列】从java虚拟机字节码执行引擎的执行过程来彻底理解java的多态性
我们知道面向对象语言的三大特点之一就是多态性,而java作为一种面向对象的语言,自然也满足多态性,我们也知道java中的多态包括重载与重写,我们也知道在C++中动态多态是通过虚函数来实现的,而虚函数是 ...
- Java虚拟机系列(五)---垃圾收集(GC)
Java语言最大的优势除了它的平台无关性之外,还有它的自动内存分配和垃圾收集技术,本节我先来总结一下垃圾收集相关的内容.本文将从解答以下三个问题的角度来展开: 1.哪些内存需要回收? 2.什么时候回收 ...
- 深入理解JAVA虚拟机(内存模型+GC算法+JVM调优)
目录 1.Java虚拟机内存模型 1.1 程序计数器 1.2 Java虚拟机栈 局部变量 1.3 本地方法栈 1.4 Java堆 1.5 方法区(永久区.元空间) 附图 2.JVM内存分配参数 2.1 ...
- 谈谈如何来查看GC日志
一.首先来看一下JVM中的GC有哪几种类型? 1.-XX:UseSerialGC 虚拟机运行在Client模式的默认值,打开此开关参数后,使用Serial+Serial Old收集器组合进行垃圾收集. ...
随机推荐
- 二分查找总结及部分Lintcode题目分析 4
二分法不只能像之前的记录,可以找到index~第二种类型是找到二分答案.有以下几个例子,都是之前二分法的扩展,不再赘述,只记录下忽略的点,以后回顾多注意~ 1. wood cut class Solu ...
- 样本方差的抽样分布 χ2(n) 卡方分布_样本方差 卡方分布
样本方差的抽样分布 χ2(n) 卡方分布_样本方差 卡方分布 样本方差的抽样分布 χ2(n) 卡方分布 t分布.卡方分布.f分布均要求总体服从正态分布. 若n个相互独立的随机变量ξ1,ξ2,-,ξn ...
- JAXB注解使用
一.Jaxb处理java对象和xml之间转换常用的annotation有: @XmlType @XmlElement @XmlRootElement @XmlAttribute @XmlAccesso ...
- SQLAlchemy的out join
我有一个app表,一个usergroup表,还有一个app_access_map表.用以实现对app访问的白名单控制. app和usergroup是多对多关系,app_access_map是中间表,不 ...
- 初识OpenCV-Python - 008: 形态转换
本节学习了图片的形态转换,即利用函数和图像的前景色和背景色去侵蚀或者扩张图像图形. import cv2import numpy as npfrom matplotlib import pyplot ...
- Python调用DLL动态链接库——ctypes使用
最近要使用python调用C++编译生成的DLL动态链接库,因此学习了一下ctypes库的基本使用. ctypes是一个用于Python的外部函数库,它提供C兼容的数据类型,并允许在DLL或共享库中调 ...
- java笔试之完全数计算
完全数(Perfect number),又称完美数或完备数,是一些特殊的自然数. 它所有的真因子(即除了自身以外的约数)的和(即因子函数),恰好等于它本身. 例如:28,它有约数1.2.4.7.14. ...
- 使用Flume+Kafka+SparkStreaming进行实时日志分析
每个公司想要进行数据分析或数据挖掘,收集日志.ETL都是第一步的,今天就讲一下如何实时地(准实时,每分钟分析一次)收集日志,处理日志,把处理后的记录存入Hive中,并附上完整实战代码 1. 整体架构 ...
- Python全栈开发:pymysql
本篇对于Python操作MySQL主要使用两种方式: 原生模块 pymsql ORM框架 SQLAchemy pymsql pymsql是Python中操作MySQL的模块,其使用方法和MySQLdb ...
- 阿里云容器服务通过LoadBalancer暴露IPv6服务
背景: IPv4地址已接近枯竭,被誉为下一代互联网技术的IPv6成为新的“全球互联网门牌号”,它可以让地球上的每一粒沙子都拥有地址.当下,各国都在加速推进下一代互联网的部署,工信部也互联网服务商提出了 ...