JVM-11虚拟机性能监控与故障处理工具之【JDK的可视化工具-JConsole】

简介: JVM-11虚拟机性能监控与故障处理工具之【JDK的可视化工具-JConsole】

思维导图

20180807073036703.png

概述


JVM-10虚拟机性能监控与故障处理工具之【命令行】我们接触了JDK提供的命令行工具,JDK还为我们提供了两个功能强大的可视化工具:JConsole和VisualVM。


JConsole在JDK1.5版本供就已经提供,而VisualVM是在JDK1.6 Update7中才首次发布。现在已经成为Oracle主力推动的多合一故障处理工具。


JConsole: Java监视与管理平台


JConsole ( Java Monitoring and Management Console)是一种基于JMX的可视化监控、管理工具。 它管理部分的功能是针对JMX MBean进行管理。 由于MBean可以在代码、中间件服务器的管理控制台或者所有符合JMX规范的软件进行访问。


所以我们这里重点介绍JConsole监视部分的功能。


启动jconsole


通过JDK安装目录下的bin目录“jconsole.exe”启动JConsole后,会自动列出本机运行的所有虚拟机进程,无需通过jps来查询了。除了本地进程,也可以使用“远程进程”的共鞥来连接远程服务器。


image.png


内存监控示例

VM ARGS

-Xms100m -Xmx100m -XX:+UseSerialGC -verbose:gc -XX:+PrintGCDetails


image.png

堆最小内存100M 最大内存100M, 即为不可扩展。 使用Serial垃圾收集器。同时为了方便和图形化界面展示的数据比对,我们打印GC日志以及堆信息


代码

package com.artisan.gc;
import java.util.ArrayList;
import java.util.List;
public class OOMObject {
  // 内存占位符对象 一个object 大约占用64kb
  public byte[] object = new byte[64 * 1024];
  public static void fillData2Heap(int number) throws InterruptedException {
    List<OOMObject> objectList = new ArrayList<OOMObject>();
    for (int i = 0; i < number; i++) {
      // 休眠50毫秒,使曲线变化的更加明显
      Thread.sleep(50);
      objectList.add(new OOMObject());
    }
    System.gc();
  }
  public static void main(String[] args) throws InterruptedException {
    fillData2Heap(1000);
    // 为了让JConsole上的内存变化体现在工具上,让主线程不要立刻退出,休眠10S
    Thread.sleep(10 * 1000);
  }
}


JConsole监控展示及说明

运行程序,执行JConsole.exe .


20180807144109320.png



上图为整个堆内存的使用情况,可以看到随着程序的运行,整个堆中内存使用是一个平滑向上增长的曲线(平的部分为Thread.sleep(10 * 1000)造成的,)程序运行结束退出后,内存归为0。

然后我们我们来看下Eden区域的曲线变化情况,如下图。


20180807151907864.png

可以很明显的看到发生了2次 Minor GC. 和输出的GC日志信息一致。


20180807151352908.png

同时在这里也可以看出来

2018080715205951.png


新生代主要是复制算法,这里的统计信息也是 2次收集。


扩展问题

没有指定-Xmn,如何确定新生代和Eden的大小

2018080714555227.png


可以从如下图中查看每个分区的内存分配情况


20180807152247951.png


当然也可以粗略计算出来


默认的,新生代 ( Young ) 与老年代 ( Old ) 的比例的值为 1:2 ( 该值可以通过参数 –XX:NewRatio 来指定 ),即:新生代 ( Young ) = 1/3 的堆空间大小。老年代 ( Old ) = 2/3 的堆空间大小。


堆内存为100M,采用默认的–XX:NewRatio,可以推算出 新生代占用1/3 , 33M .


没有指定-XX:SurvivorRatio , 默认为8。


新生代又分为 Eden区、SurvivorFrom、SurvivorTo三个区


默认的,Edem : from : to = 8 : 1 : 1 ( 可以通过参数 –XX:SurvivorRatio 来设定 ),即: Eden = 8/10 的新生代空间大小,from = to = 1/10 的新生代空间大小。


JVM 每次只会使用 Eden 和其中的一块 Survivor 区域来为对象服务,所以无论什么时候,总是有一块 Survivor 区域是空闲着的。


因此,新生代实际可用的内存空间为 9/10 ( 即90% )的新生代空间,只有10%的内存被“浪费”,最大限度的节约资源。


Eden内存区的大小为 (33M * 9/10 ) 新生代可用的空间 * 8/9 (Eden占用新生代的比例) = 27033.6KB 。 和 图中显示的27328 KB 接近。


为何老年代的柱状图信息仍显示峰值状态,如何调整代码回收该区域

20180807153248664.png


可以看到老年代仍然占据很大的内存空间,是因为 虽然执行了System.gc()。 但是是在fillData2Heap方法内执行的。 在方法内,List<OOMObject>是存活的,该方法还没退出,gc无法回收。


只要在该方法外执行GC就可以将老年代的内存回收。 我们来实践下

20180807153749417.png


启动JConsole


20180807153943741.png

线程监控示例


上面的“内存”可以看做是 jstat命令的话,“线程”页签就可相当于jstack命令, 遇到线程停顿时可以使用这个页签进行监控分析。 用于定位线程长时间停顿的原因,比如等待外部资源(数据库连接、网络资源、设备资源等),锁等待(活锁和死锁)、死循环。


活锁等待示例


package com.artisan.gc;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
public class ThreadWaitTest {
  /**
   * 
   * 
   * @Title: createBusyThread
   * 
   * @Description: 线程死循环演示
   * 
   * 
   * @return: void
   */
  public static void createBusyThread() {
    Thread thread  = new Thread(new Runnable() {
      @Override
      public void run() {
        while (true) {
          // do something
          // System.out.println("busyThread running");
        }
      }
    }, "busyThread");
    // 启动线程
    thread.start();
  }
  /**
   * 
   * 
   * @Title: createLockThread
   * 
   * @Description: 线程锁等待演示
   * 
   * @param lock
   * 
   * @return: void
   */
  public static void createLockThread(final Object lock) {
    Thread  thread = new Thread(new Runnable() {
      @Override
      public void run() {
        synchronized (lock) {
          try {
            lock.wait();
          } catch (InterruptedException e) {
            e.printStackTrace();
          }
        }
      }
    }, "lockThread");
    thread.start();
  }
  public static void main(String[] args) throws IOException {
    System.out.println("请输入");
    BufferedReader bReader = new BufferedReader(new InputStreamReader(System.in));
    bReader.readLine();
    createBusyThread();
    bReader.readLine();
    Object object = new Object();
    createLockThread(object);
  }
}


运行程序,启动 jconsole, 选择该进程 ,选择 “线程”页签,

首先查看main线程


20180807185215207.png


堆栈追踪显示


20180807185830951.png


BufferedReader在readBytes方法中等待System.in的键盘输入,这时线程状态为Runnable状态,Runnable状态的线程会被分配运行时间,但readBytes方法检测到流没有更新会立刻归还执行令牌,这种等待只消耗很小的cpu资源。


输入内容后,接着监控 “busyThread”线程


20180807185653680.png


可以看出一直在ThreadWaitTest.java的 第24行停留 ,我们查看ThreadWaitTest的24行内容


20180807190208367.png

while (true)

这时线程为Runnable状态,而且没有归还线程执行令牌的动作,会在空循环上用尽全部的执行时间,直到线程切换,这种等待会消耗较多的CPU资源。

接着控制台再输入内容


20180807190742278.png

继续回到jconsole查看 “lockThread”线程


20180807190829560.png


lockThread线程正在处于正常的活锁等待,只要lock对象的notify或者notifyAll方法被调用,这个线程便可以继续执行。


死锁等待示例

package com.artisan.gc;
public class ThreadLockTest {
  static class ThreadDemo implements Runnable {
    int a, b;
    public ThreadDemo(int a, int b) {
      this.a = a;
      this.b = b;
    }
    @Override
    public void run() {
      synchronized (Integer.valueOf(a)) {
        synchronized (Integer.valueOf(b)) {
          System.out.println(a + b);
        }
      }
    }
  }
  public static void main(String[] args) {
    // 加入循环,是为了更高概率的造成死锁
    for (int i = 0; i < 200; i++) {
      new Thread(new ThreadDemo(1, 2)).start();
      new Thread(new ThreadDemo(2, 1)).start();
    }
  }
}


先分析下造成死锁的原因: Integer.valueOf()方法基于减少对象创建次数和节省内存的考虑,【-128,127】之间的数字会被缓存,当valueOf()方法传入参数在这个范围内的时候,将直接返回缓存中的对象。 也就是说上述代码中虽然调用了200次的Integer.valueOf()方法,但是入参为1和2,一共就返回了2个不同的对象。 假如在某个线程的两个synchronized块之间发生了一次线程切换,那么就会出现线程A等着被线程B持有的Integer.valueOf(1),线程B等着被线程A持有的Integer.valueOf(2),造成死锁。


点击“检测到死锁”按钮,如果 有死锁将会出现一个新的“死锁”页签。


201808071917393.png


image.png


image.png


可以看到Thread-27和Thread-28互相阻塞,已经没有等到锁释放的希望了。

相关文章
|
监控 Java Unix
6个Java 工具,轻松分析定位 JVM 问题 !
本文介绍了如何使用 JDK 自带工具查看和分析 JVM 的运行情况。通过编写一段测试代码(启动 10 个死循环线程,分配大量内存),结合常用工具如 `jps`、`jinfo`、`jstat`、`jstack`、`jvisualvm` 和 `jcmd` 等,详细展示了 JVM 参数配置、内存使用、线程状态及 GC 情况的监控方法。同时指出了一些常见问题,例如参数设置错误导致的内存异常,并通过实例说明了如何排查和解决。最后附上了官方文档链接,方便进一步学习。
2979 4
|
数据挖掘 索引
服务器数据恢复—raid6阵列硬盘故障导致上层虚拟机不可用的数据恢复案例
一台由16块硬盘组成的raid6磁盘阵列。磁盘阵列中有一块硬盘因为物理故障掉线,导致服务器上层虚拟机无法正常使用,部分分区丢失,重启物理服务器后发现数据丢失。
|
存储 数据挖掘 虚拟化
vsan数据恢复—vsan缓存盘故障导致虚拟机磁盘文件丢失的数据恢复案例
VMware vsan架构采用2+1模式。每台设备只有一个磁盘组(7+1),缓存盘的大小为240GB,容量盘的大小为1.2TB。 由于其中一台主机(0号组设备)的缓存盘出现故障,导致VMware虚拟化环境中搭建的2台虚拟机的磁盘文件(vmdk)丢失。
|
JavaScript 前端开发 Java
jvm的jshell,学生的工具
本文介绍了JVM的jshell工具,它为Java平台添加了REPL(读取-评估-打印循环)功能,使得学习、探索编码和原型代码变得更加便捷,但作者认为其在实际开发中较为鸡肋。
205 2
jvm的jshell,学生的工具
|
Arthas Prometheus 监控
监控堆外使用JVM工具
监控堆外使用JVM工具
549 7
|
Arthas 监控 数据可视化
JVM进阶调优系列(7)JVM调优监控必备命令、工具集合|实用干货
本文介绍了JVM调优监控命令及其应用,包括JDK自带工具如jps、jinfo、jstat、jstack、jmap、jhat等,以及第三方工具如Arthas、GCeasy、MAT、GCViewer等。通过这些工具,可以有效监控和优化JVM性能,解决内存泄漏、线程死锁等问题,提高系统稳定性。文章还提供了详细的命令示例和应用场景,帮助读者更好地理解和使用这些工具。
|
存储 运维 虚拟化
虚拟化数据恢复——Hyper-V虚拟化故障导致虚拟机文件丢失的数据恢复案例
在Windows Server上部署的Hyper-V虚拟化环境中,因存储中虚拟机数据文件丢失导致服务瘫痪。北亚企安数据恢复工程师通过物理检测、操作系统及文件系统检测,确定为人为格式化造成,并通过镜像硬盘、重组RAID、分析并恢复文件索引项等步骤,成功恢复数据,最终在新Hyper-V环境中验证并迁移所有虚拟机,确保用户业务恢复正常运行。
|
存储 监控 算法
JVM调优深度剖析:内存模型、垃圾收集、工具与实战
【10月更文挑战第9天】在Java开发领域,Java虚拟机(JVM)的性能调优是构建高性能、高并发系统不可或缺的一部分。作为一名资深架构师,深入理解JVM的内存模型、垃圾收集机制、调优工具及其实现原理,对于提升系统的整体性能和稳定性至关重要。本文将深入探讨这些内容,并提供针对单机几十万并发系统的JVM调优策略和Java代码示例。
489 2
|
小程序 Oracle Java
JVM知识体系学习一:JVM了解基础、java编译后class文件的类结构详解,class分析工具 javap 和 jclasslib 的使用
这篇文章是关于JVM基础知识的介绍,包括JVM的跨平台和跨语言特性、Class文件格式的详细解析,以及如何使用javap和jclasslib工具来分析Class文件。
424 0
JVM知识体系学习一:JVM了解基础、java编译后class文件的类结构详解,class分析工具 javap 和 jclasslib 的使用
|
Arthas 存储 算法
深入理解JVM,包含字节码文件,内存结构,垃圾回收,类的声明周期,类加载器
JVM全称是Java Virtual Machine-Java虚拟机JVM作用:本质上是一个运行在计算机上的程序,职责是运行Java字节码文件,编译为机器码交由计算机运行类的生命周期概述:类的生命周期描述了一个类加载,使用,卸载的整个过类的生命周期阶段:类的声明周期主要分为五个阶段:加载->连接->初始化->使用->卸载,其中连接中分为三个小阶段验证->准备->解析类加载器的定义:JVM提供类加载器给Java程序去获取类和接口字节码数据类加载器的作用:类加载器接受字节码文件。
986 55