JVM-07垃圾收集Garbage Collection【GC日志分析】

本文涉及的产品
日志服务 SLS,月写入数据量 50GB 1个月
简介: JVM-07垃圾收集Garbage Collection【GC日志分析】

2018073021082440.png2018073021082440.png概述

每一种收集器的日志形式都是由他们自身的实现决定的,也就是说每个收集器的日志格式都可能不一样。


GC格式

一般来说,常见的格式如下:

[GC [<collector>: <starting occupancy1> -> <ending occupancy1>, <pause time1> secs] <starting occupancy3> -> <ending occupancy3>, <pause time3> secs]


<collector> GC为minor收集过程中使用的垃圾收集器起的内部名称.

<starting occupancy1> young generation 在进行垃圾收集前被对象使用的存储空间.

<ending occupancy1> young generation 在进行垃圾收集后被对象使用的存储空间

<pause time1> minor收集使应用暂停的时间长短(秒)

<starting occupancy3> 整个堆(Heap Size)在进行垃圾收集前被对象使用的存储空间

<ending occupancy3> 整个堆(Heap Size)在进行垃圾收集后被对象使用的存储空间

<pause time3> 整个垃圾收集使应用暂停的时间长短(秒),包括major收集使应用暂停的时间(如果发生了major收集).


实例

我们用前面博文中的例子来打印下GC日志,然后分析解读下

VM Args :

-XX:+PrintGCDetails  -XX:+PrintGCApplicationConcurrentTime  -XX:+PrintGCApplicationStoppedTime

2018073021082440.png


  • -XX:+PrintGCDetails 显示GC的详细信息
  • -XX:+PrintGCApplicationConcurrentTime 打印应用执行的时间
  • -XX:+PrintGCApplicationStoppedTime 打印应用被暂停的时间
package com.artisan.gc;
/**
 * 
 * 
 * @ClassName: ReferenceCountingGC
 * 
 * @Description: VM Args
 * 
 * @author: Mr.Yang
 * 
 * @date: 2018年7月29日 上午10:31:32
 */
public class ReferenceCountingGC {
  private Object instance;
  private static final int _1M = 1024 * 1024;
  // 设置个成员变量,在堆中占点内存,以便观察GC是否回收相互引用的情况
  private byte[] bigByte = new byte[2 * _1M];
  public static void main(String[] args) {
    ReferenceCountingGC rc = new ReferenceCountingGC();
    ReferenceCountingGC rc2 = new ReferenceCountingGC();
    // 设置相互引用
    rc.instance = rc2;
    rc2.instance = rc;
    // 将对象置为空
    rc.instance = null;
    rc2.instance = null;
    // 垃圾回收,观察rc 和 rc2能否被回收
    System.gc();
  }
}

设置完虚拟机参数后,执行main方法,输入如下

Application time: 0.0087943 seconds
[GC [PSYoungGen: 4906K->4400K(47232K)] 4906K->4400K(155264K), 0.0020510 secs] [Times: user=0.00 sys=0.00, real=0.00 secs] 
[Full GC (System) [PSYoungGen: 4400K->0K(47232K)] [PSOldGen: 0K->4255K(108032K)] 4400K->4255K(155264K) [PSPermGen: 2977K->2977K(21248K)], 0.0076827 secs] [Times: user=0.01 sys=0.00, real=0.01 secs] 
Total time for which application threads were stopped: 0.0098759 seconds
Heap
 PSYoungGen      total 47232K, used 1620K [0x00000007cb400000, 0x00000007ce8b0000, 0x0000000800000000)
  eden space 40512K, 4% used [0x00000007cb400000,0x00000007cb5952b8,0x00000007cdb90000)
  from space 6720K, 0% used [0x00000007cdb90000,0x00000007cdb90000,0x00000007ce220000)
  to   space 6720K, 0% used [0x00000007ce220000,0x00000007ce220000,0x00000007ce8b0000)
 PSOldGen        total 108032K, used 4255K [0x0000000761c00000, 0x0000000768580000, 0x00000007cb400000)
  object space 108032K, 3% used [0x0000000761c00000,0x0000000762027c80,0x0000000768580000)
 PSPermGen       total 21248K, used 2995K [0x000000075ca00000, 0x000000075dec0000, 0x0000000761c00000)
  object space 21248K, 14% used [0x000000075ca00000,0x000000075ccecc30,0x000000075dec0000)
Application time: 0.0006608 seconds



我们重点来说下 -XX:+PrintGCDetails 虚拟机参数输出的数据


GC日志开头的 [GC 和 [Full GC 说明了这次垃圾收集器的停顿类型,而不是用来区分新生代GC还是老年GC的。


如果有 Full ,说明了这次GC发生了Stop-The-World.


如果是调用了System.gc()方法所触发的收集,那么将显示[Full GC (System),如上日志所示。


接下来的“[DefNew” 或者 “[Tenured” 或者“[Perm” 表示GC发生的区域,这里显示的区域名称和使用的GC收集器密切相关。


如果是用的Serial收集器,新生代名为“Default New Generation”,所以显示“[DefNew”。

如果是用的ParNew收集器,新生代名为“Parallel New Generation”,所以显示“[ParNew”。


如果是用的Parallel Scavenge收集器,新生代名则显示为“[PSYongGen”,如日志中显示。

老年代和永久代同理,名称也是由收集器决定的。


后面方括号内部的 4906K->4400K(47232K)含义是 “GC前该内存区域已经使用的容量->GC后该内存区域已使用的容量(该内存区域中容量)”


方括号之外的 4906K->4400K(155264K) 表示“GC前Java堆已使用容量->GC后Java堆已使用容量(Java堆中容量)”


再往后 0.0020510 secs表示该内存区域GC所占用的时间,单位为秒。

[Times: user=0.00 sys=0.00, real=0.00 secs]


是更具体的时间数据,取决于收集器,有些收集器可能不打印。


这里面的user、sys、real和Linux的time命令输出的时间含义是一致的,分别代表用户态消耗的CPU时间、内核态消耗的CPU时间和操作从开始到结束锁经理的墙钟时间(Wall Clock Time).


CPU时间和墙钟时间的区别是: 墙钟时间包括各种费运算的等待耗时,比如磁盘I/O、等待线程阻塞。 而CPU时间则不包含这些耗时,但当系统有多CPU或者多核的话,多线程操作会叠加CPU时间,所以读者看到user或者sys时间超过real时间是完全正常的。


GC监控


##打开 Java VisualVM

市面上有很多GC监控和分析工具,这里我们使用JDK自带的Java VisualVM,位于JDK的bin目录下



image.png


Java VisualVM的作用不止于此:

  • 生成和分析堆内存转储
  • 查看和操作MBean
  • 监控垃圾回收
  • 内存和CPU分析


安装GC可视化插件


工具–插件


2018073119225159.png


启动本地进程,监控和分析GC情况


20180731192440319.png


20180731192529309.png


还有些其他功能,这里不再阐述了,如何分析观察,另开篇找个案例介绍。


GC日志分析工具


如果GC日志比较多,一般情况我们不直接阅读,都是通过GC日志分析工具提供的图形化页面来查看。

这里省略,后续开篇单独介绍

有个网页版的 http://gceasy.io/

20180731192902298.png


20180731193031701.png


相关实践学习
日志服务之使用Nginx模式采集日志
本文介绍如何通过日志服务控制台创建Nginx模式的Logtail配置快速采集Nginx日志并进行多维度分析。
相关文章
|
1月前
|
缓存 算法 Java
JVM知识体系学习六:JVM垃圾是什么、GC常用垃圾清除算法、堆内存逻辑分区、栈上分配、对象何时进入老年代、有关老年代新生代的两个问题、常见的垃圾回收器、CMS
这篇文章详细介绍了Java虚拟机(JVM)中的垃圾回收机制,包括垃圾的定义、垃圾回收算法、堆内存的逻辑分区、对象的内存分配和回收过程,以及不同垃圾回收器的工作原理和参数设置。
65 4
JVM知识体系学习六:JVM垃圾是什么、GC常用垃圾清除算法、堆内存逻辑分区、栈上分配、对象何时进入老年代、有关老年代新生代的两个问题、常见的垃圾回收器、CMS
|
20天前
|
存储 监控 Java
JVM进阶调优系列(8)如何手把手,逐行教她看懂GC日志?| IT男的专属浪漫
本文介绍了如何通过JVM参数打印GC日志,并通过示例代码展示了频繁YGC和FGC的场景。文章首先讲解了常见的GC日志参数,如`-XX:+PrintGCDetails`、`-XX:+PrintGCDateStamps`等,然后通过具体的JVM参数和代码示例,模拟了不同内存分配情况下的GC行为。最后,详细解析了GC日志的内容,帮助读者理解GC的执行过程和GC处理机制。
|
1月前
|
存储 监控 算法
JVM调优深度剖析:内存模型、垃圾收集、工具与实战
【10月更文挑战第9天】在Java开发领域,Java虚拟机(JVM)的性能调优是构建高性能、高并发系统不可或缺的一部分。作为一名资深架构师,深入理解JVM的内存模型、垃圾收集机制、调优工具及其实现原理,对于提升系统的整体性能和稳定性至关重要。本文将深入探讨这些内容,并提供针对单机几十万并发系统的JVM调优策略和Java代码示例。
51 2
|
1月前
|
Arthas 监控 Java
JVM知识体系学习七:了解JVM常用命令行参数、GC日志详解、调优三大方面(JVM规划和预调优、优化JVM环境、JVM运行出现的各种问题)、Arthas
这篇文章全面介绍了JVM的命令行参数、GC日志分析以及性能调优的各个方面,包括监控工具使用和实际案例分析。
46 3
|
1月前
|
算法 Java
JVM进阶调优系列(4)年轻代和老年代采用什么GC算法回收?
本文详细介绍了JVM中的GC算法,包括年轻代的复制算法和老年代的标记-整理算法。复制算法适用于年轻代,因其高效且能避免内存碎片;标记-整理算法则用于老年代,虽然效率较低,但能有效解决内存碎片问题。文章还解释了这两种算法的具体过程及其优缺点,并简要提及了其他GC算法。
 JVM进阶调优系列(4)年轻代和老年代采用什么GC算法回收?
|
16天前
|
XML 安全 Java
【日志框架整合】Slf4j、Log4j、Log4j2、Logback配置模板
本文介绍了Java日志框架的基本概念和使用方法,重点讨论了SLF4J、Log4j、Logback和Log4j2之间的关系及其性能对比。SLF4J作为一个日志抽象层,允许开发者使用统一的日志接口,而Log4j、Logback和Log4j2则是具体的日志实现框架。Log4j2在性能上优于Logback,推荐在新项目中使用。文章还详细说明了如何在Spring Boot项目中配置Log4j2和Logback,以及如何使用Lombok简化日志记录。最后,提供了一些日志配置的最佳实践,包括滚动日志、统一日志格式和提高日志性能的方法。
137 30
【日志框架整合】Slf4j、Log4j、Log4j2、Logback配置模板
|
1月前
|
XML JSON Java
Logback 与 log4j2 性能对比:谁才是日志框架的性能王者?
【10月更文挑战第5天】在Java开发中,日志框架是不可或缺的工具,它们帮助我们记录系统运行时的信息、警告和错误,对于开发人员来说至关重要。在众多日志框架中,Logback和log4j2以其卓越的性能和丰富的功能脱颖而出,成为开发者们的首选。本文将深入探讨Logback与log4j2在性能方面的对比,通过详细的分析和实例,帮助大家理解两者之间的性能差异,以便在实际项目中做出更明智的选择。
234 3
|
3月前
|
Kubernetes Ubuntu Windows
【Azure K8S | AKS】分享从AKS集群的Node中查看日志的方法(/var/log)
【Azure K8S | AKS】分享从AKS集群的Node中查看日志的方法(/var/log)
135 3
|
1月前
|
存储 缓存 关系型数据库
MySQL事务日志-Redo Log工作原理分析
事务的隔离性和原子性分别通过锁和事务日志实现,而持久性则依赖于事务日志中的`Redo Log`。在MySQL中,`Redo Log`确保已提交事务的数据能持久保存,即使系统崩溃也能通过重做日志恢复数据。其工作原理是记录数据在内存中的更改,待事务提交时写入磁盘。此外,`Redo Log`采用简单的物理日志格式和高效的顺序IO,确保快速提交。通过不同的落盘策略,可在性能和安全性之间做出权衡。
1638 14
|
1月前
|
Python
log日志学习
【10月更文挑战第9天】 python处理log打印模块log的使用和介绍
35 0
下一篇
无影云桌面