javap: 反编译工具, 可用来查看java编译器生成的字节码

参数摘要:

  • -help 帮助
  • -l 输出行和变量的表
  • -public 只输出public方法和域
  • -protected 只输出public和protected类和成员
  • -package 只输出包,public和protected类和成员,这是默认的
  • -p -private 输出所有类和成员
  • -s 输出内部类型签名
  • -c 输出分解后的代码,例如,类中每一个方法内,包含java字节码的指令
  • -verbose 输出栈大小,方法参数的个数
  • -constants 输出静态final常量

代码清单:

import java.awt.*;
import java.applet.*; public class DocFooter extends Applet {
String date;
String email; public void init() {
resize(500,100);
date = getParameter("LAST_UPDATED");
email = getParameter("EMAIL");
} public void paint(Graphics g) {
g.drawString(date + " by ",100, 15);
g.drawString(email,290,15);
}
}

命令行下输入javap DocFooter, 结果如下:

D:\cn\javaCommand>javap DocFooter
Compiled from "DocFooter.java"
public class DocFooter extends java.applet.Applet {
java.lang.String date;
java.lang.String email;
public DocFooter();
public void init();
public void paint(java.awt.Graphics);
}

jinfo: Java配置信息工具

惯例, 先启动一个java应用程序, 通过jps命令查看java进程的pid为, 命令行下输入jinfo -J-d64 -sysprops 14568。这里的 -J-d64 参数对应64位虚拟机。

C:\Users\liqingshan>jinfo -J-d64 -sysprops 14568
Attaching to process ID 14568, please wait...
Debugger attached successfully.
Server compiler detected.
JVM version is 25.161-b12
java.runtime.name = Java(TM) SE Runtime Environment
java.vm.version = 25.161-b12
sun.boot.library.path = E:\javaTools\jre\bin
java.vendor.url = http://java.oracle.com/
java.vm.vendor = Oracle Corporation
path.separator = ;
file.encoding.pkg = sun.io
java.vm.name = Java HotSpot(TM) 64-Bit Server VM
sun.os.patch.level =
sun.java.launcher = SUN_STANDARD
user.script =
user.country = CN
user.dir = D:\cn\javaCommand
java.vm.specification.name = Java Virtual Machine Specification
java.runtime.version = 1.8.0_161-b12
java.awt.graphicsenv = sun.awt.Win32GraphicsEnvironment
os.arch = amd64
java.endorsed.dirs = E:\javaTools\jre\lib\endorsed
line.separator = java.io.tmpdir = C:\Users\LIQING~1\AppData\Local\Temp\
java.vm.specification.vendor = Oracle Corporation
user.variant =
os.name = Windows 10
sun.jnu.encoding = GBK
java.library.path = C:\ProgramData\Oracle\Java\javapath;C:\WINDOWS\Sun\Java\bin;C:\WINDOWS\system32;C:\WINDOWS;C:\ProgramData\Oracle\Java\javapath;C:\oracle\product\10.2.0\db_1\bin;E:\javaTools\jdk1.8.0_161\bin;C:\Program Files (x86)\Intel\iCLS Client\;C:\Program Files\Intel\iCLS Client\;C:\windows\system32;C:\windows;C:\windows\System32\Wbem;C:\windows\System32\WindowsPowerShell\v1.0\;C:\Program Files (x86)\Intel\Intel(R) Management Engine Components\DAL;C:\Program Files\Intel\Intel(R) Management Engine Components\DAL;C:\Program Files (x86)\Intel\Intel(R) Management Engine Components\IPT;C:\Program Files\Intel\Intel(R) Management Engine Components\IPT;C:\Program Files (x86)\NVIDIA Corporation\PhysX\Common;c:\Program Files\Intel\WiFi\bin\;C:\Java\MySQL\MySQL Server 5.5\bin;c:\Program Files\Common Files\Intel\WirelessCommon\;C:\Java\VisualSVN Server\bin;C:\Java\apache-maven-3.3.9\bin;E:\javaTools\apache-cxf-3.1.9\bin;E:\javaTools\Git\cmd;E:\javaTools\TortoiseGit\bin;E:\javaTools\mongodb\bin;C:\WINDOWS\system32;C:\WINDOWS;C:\WINDOWS\System32\Wbem;C:\WINDOWS\System32\WindowsPowerShell\v1.0\;C:\Program Files\Intel\WiFi\bin\;C:\Program Files\Common Files\Intel\WirelessCommon\;C:\WINDOWS\System32\OpenSSH\;C:\Program Files\TortoiseSVN\bin;E:\javaTools\nodejs\;E:\Program Files\Bandizip\;C:\Users\liqingshan\AppData\Local\Microsoft\WindowsApps;C:\Users\liqingshan\AppData\Roaming\npm;.
java.specification.name = Java Platform API Specification
java.class.version = 52.0
sun.management.compiler = HotSpot 64-Bit Tiered Compilers
os.version = 10.0
user.home = C:\Users\liqingshan
user.timezone =
java.awt.printerjob = sun.awt.windows.WPrinterJob
file.encoding = GBK
java.specification.version = 1.8
user.name = qingshan
java.class.path = .
java.vm.specification.version = 1.8
sun.arch.data.model = 64
sun.java.command = JpsDemo
java.home = E:\javaTools\jre
user.language = zh
java.specification.vendor = Oracle Corporation
awt.toolkit = sun.awt.windows.WToolkit
java.vm.info = mixed mode
java.version = 1.8.0_161
java.ext.dirs = E:\javaTools\jre\lib\ext;C:\WINDOWS\Sun\Java\lib\ext
sun.boot.class.path = E:\javaTools\jre\lib\resources.jar;E:\javaTools\jre\lib\rt.jar;E:\javaTools\jre\lib\sunrsasign.jar;E:\javaTools\jre\lib\jsse.jar;E:\javaTools\jre\lib\jce.jar;E:\javaTools\jre\lib\charsets.jar;E:\javaTools\jre\lib\jfr.jar;E:\javaTools\jre\classes
java.vendor = Oracle Corporation
sun.stderr.encoding = ms936
file.separator = \
java.vendor.url.bug = http://bugreport.sun.com/bugreport/
sun.io.unicode.encoding = UnicodeLittle
sun.cpu.endian = little
sun.stdout.encoding = ms936
sun.desktop = windows
sun.cpu.isalist = amd64

另外网上找到的相关博客以及官方文档都指明 JDK 1.8 之后已经不再支持jinfo命令, 但是在我本地 1.8 环境下却支持, 这个先在这里做个记录, 待以后找到原因再完善本篇文章。

jmap: 生成heapdump文件

JDK的命令行工具系列 (一) jps、jstat中, 我们已经介绍过heapdump文件, 所以这里就直接对jmap的命令和参数做个演示。其中关于具体参数的摘要说明转载自H大的博客:  Java命令学习系列(三)——Jmap-HollisChuang's Blog

命令格式

  jmap [option] <pid>
    (to connect to running process)
  jmap [option] <executable <core>
    (to connect to a core file)
  jmap [option] [server_id@]<remote server IP or hostname>
    (to connect to remote debug server)

参数摘要

  • option 选项参数是互斥的(不可同时使用)。想要使用选项参数,直接跟在命令名称后即可。
  • pid 需要打印配置信息的进程ID。该进程必须是一个Java进程。想要获取运行的Java进程列表,你可以使用jps。
  • executable 产生核心dump的Java可执行文件。
  • core 需要打印配置信息的核心文件。
  • remote server IP or hostname 远程调试服务器的(请查看jsadebugd)主机名或IP地址。
  • server-id 可选的唯一id,如果相同的远程主机上运行了多台调试服务器,用此选项参数标识服务器。

option

  • <no option> 如果使用不带选项参数的jmap打印共享对象映射,将会打印目标虚拟机中加载的每个共享对象的起始地址、映射大小以及共享对象文件的路径全称。这与Solaris的pmap工具比较相似。
  • -dump:[live,]format=b,file=<filename> 以hprof二进制格式转储Java堆到指定filename的文件中。live子选项是可选的。如果指定了live子选项,堆中只有活动的对象会被转储。想要浏览heap dump,你可以使用jhat(Java堆分析工具)读取生成的文件。
  • -finalizerinfo 打印等待终结的对象信息。
  • -heap 打印一个堆的摘要信息,包括使用的GC算法、堆配置信息和generation wise heap usage。
  • -histo[:live] 打印堆的柱状图。其中包括每个Java类、对象数量、内存大小(单位:字节)、完全限定的类名。打印的虚拟机内部的类名称将会带有一个’*’前缀。如果指定了live子选项,则只计算活动的对象。
  • -permstat 打印Java堆内存的永久保存区域的类加载器的智能统计信息。对于每个类加载器而言,它的名称、活跃度、地址、父类加载器、它所加载的类的数量和大小都会被打印。此外,包含的字符串数量和大小也会被打印。
  • -F 强制模式。如果指定的pid没有响应,请使用jmap -dump或jmap -histo选项。此模式下,不支持live子选项。
  • -h 打印帮助信息。
  • -help 打印帮助信息。
  • -J<flag> 指定传递给运行jmap的JVM的参数。

查看java堆的摘要信息: jmap -heap 11392

C:\Users\liqingshan>jps -m
JpsDemo
12816
14352 Bootstrap start
6392 Jps -m
10060 Launcher C:/Java/IDEA/lib/guava-21.0.jar;C:/Java/IDEA/lib/jps-builders-6.jar;C:/Java/IDEA/lib/log4j.jar;C:/Java/IDEA/lib/httpcore-4.4.5.jar;C:/Java/IDEA/lib/netty-all-4.1.13.Final.jar;C:/Java/IDEA/lib/commons-codec-1.9.jar;C:/Java/IDEA/lib/annotations.jar;C:/Java/IDEA/lib/snappy-in-java-0.5.1.jar;C:/Java/IDEA/lib/trove4j.jar;C:/Java/IDEA/lib/jdom.jar;C:/Java/IDEA/lib/httpclient-4.5.2.jar;C:/Java/IDEA/lib/util.jar;C:/Java/IDEA/lib/openapi.jar;C:/Java/IDEA/lib/jps-model.jar;C:/Java/IDEA/lib/aether-dependency-resolver.jar;C:/Java/IDEA/lib/oromatcher.jar;C:/Java/IDEA/lib/forms_rt.jar;C:/Java/IDEA/lib/idea_rt.jar;C:/Java/IDEA/lib/resources_en.jar;C:/Java/IDEA/lib/protobuf-java-2.5.0.jar;C:/Java/IDEA/lib/asm-all.jar;C:/Java/IDEA/lib/jna.jar;C:/Java/IDEA/lib/jgoodies-forms.jar;C:/Java/IDEA/lib/javac2.jar;C:/Java/IDEA/lib/jps-builders.jar;C:/Java/IDEA/lib/slf4j-api-1.7.10.jar;C:/Java/IDEA/lib/aether-1.1.0-all.jar;C:/Java/IDEA/lib/commons-logging-1.2.jar;C:/Java/IDEA/lib/jna-platform.jar;C:/J C:\Users\liqingshan>jmap -heap 11392
Attaching to process ID 11392, please wait...
Debugger attached successfully.
Server compiler detected.
JVM version is 25.161-b12 using thread-local object allocation.
Parallel GC with 4 thread(s) //GC方式 Heap Configuration: //堆内存初始化配置
MinHeapFreeRatio = 0 //对应jvm启动参数-XX:MinHeapFreeRatio设置JVM堆最小空闲比率(default 40)
MaxHeapFreeRatio = 100 //对应jvm启动参数 -XX:MaxHeapFreeRatio设置JVM堆最大空闲比率(default 70)
MaxHeapSize = 2118123520 (2020.0MB) //对应jvm启动参数-XX:MaxHeapSize=设置JVM堆的最大大小
NewSize = 44564480 (42.5MB) //对应jvm启动参数-XX:NewSize=设置JVM堆的‘新生代’的默认大小
MaxNewSize = 705691648 (673.0MB) //对应jvm启动参数-XX:MaxNewSize=设置JVM堆的‘新生代’的最大大小
OldSize = 89653248 (85.5MB) //对应jvm启动参数-XX:OldSize=<value>:设置JVM堆的‘老生代’的大小
NewRatio = 2 //对应jvm启动参数-XX:NewRatio=:‘新生代’和‘老生代’的大小比率
SurvivorRatio = 8 //对应jvm启动参数-XX:SurvivorRatio=设置年轻代中Eden区与Survivor区的大小比值
MetaspaceSize = 21807104 (20.796875MB) //对应jvm启动参数-XX:MetaspaceSize=<value>:设置JVM堆的‘元空间’的初始大小
CompressedClassSpaceSize = 1073741824 (1024.0MB) //对应jvm启动参数-XX:CompressedClassSpaceSize=<value>:设置类指针压缩空间大小
MaxMetaspaceSize = 17592186044415 MB //对应jmv启动参数-XX:MaxMetaspaceSize=<value>:设置JVM堆的'元空间'的最大大小
G1HeapRegionSize = 0 (0.0MB) Heap Usage: //堆内存使用情况
PS Young Generation
Eden Space: //Eden区内存分布
capacity = 24641536 (23.5MB) //总容量
used = 492872 (0.47003936767578125MB) //已使用容量
free = 24148664 (23.02996063232422MB) //剩余容量
2.000167522024601% used //使用比例
From Space: //其中一个Survivor区的内存分布
capacity = 524288 (0.5MB)
used = 0 (0.0MB)
free = 524288 (0.5MB)
0.0% used
To Space: //另一个Survivor区的内存分布
capacity = 524288 (0.5MB)
used = 0 (0.0MB)
free = 524288 (0.5MB)
0.0% used
PS Old Generation //当前的老年代内存分布
capacity = 34603008 (33.0MB)
used = 674280 (0.6430435180664062MB)
free = 33928728 (32.356956481933594MB)
1.9486167214133523% used 1577 interned Strings occupying 146816 bytes.

查看java堆的对象数量、内存大小、类的全限定名: jmap -histo:live 11392。其中 :live 参数会指定JVM先触发一次GC之后, 再统计堆的柱状图。

C:\Users\liqingshan>jmap -histo:live 11392

 num     #instances         #bytes  class name
----------------------------------------------
1: 2336 397136 [C
2: 497 56704 java.lang.Class
3: 2065 49560 java.lang.String
4: 792 31680 java.util.TreeMap$Entry
5: 529 30224 [Ljava.lang.Object;
6: 10 25032 [B
7: 388 9312 java.util.LinkedList$Node
8: 195 8896 [Ljava.lang.String;
9: 228 7296 java.util.HashMap$Node
10: 195 6240 java.util.LinkedList
11: 256 4096 java.lang.Integer
12: 99 3960 java.lang.ref.SoftReference
13: 117 3744 java.util.Hashtable$Entry
14: 93 3616 [I
15: 13 3088 [Ljava.util.HashMap$Node;
16: 7 2632 java.lang.Thread
17: 2 2080 [[C
18: 38 1824 sun.util.locale.LocaleObjectCache$CacheEntry
19: 51 1632 java.util.concurrent.ConcurrentHashMap$Node
20: 20 1600 java.lang.reflect.Constructor
 .................................................. // 实际测试按num编号有187条记录, 这里只贴出了前20条记录
Total 8528 674328

将Java堆的内存映像输出到指定名称的文件中: jmap -dump:format=b,file=qingshanli 11392, 然后输入jhat命令启动一个HTTP/HTML服务器: jhat -port 10000 qingshanli

C:\Users\liqingshan>jmap -dump:format=b,file=qingshanli 11392
Dumping heap to C:\Users\liqingshan\qingshanli ...
Heap dump file created C:\Users\liqingshan>jhat -port 10000 qingshanli
Reading from qingshanli...
Dump file created Mon Jul 16 19:50:18 GMT+08:00 2018
Snapshot read, resolving...
Resolving 562831 objects...
Chasing references, expect 112 dots................................................................................................................
Eliminating duplicate references................................................................................................................
Snapshot resolved.
Started HTTP server on port 10000
Server is ready.

在浏览器中输入地址: http://localhost:10000, 就可以在页面查看堆转储快照详细信息。

参考资料

Java命令学习系列(七)——javap-HollisChuang's Blog

Java命令学习系列(六)——jinfo-HollisChuang's Blog

Java命令学习系列(三)——Jmap-HollisChuang's Blog

《深入理解Java虚拟机》

作者:张小凡
出处:https://www.cnblogs.com/qingshanli/

本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。如果觉得还有帮助的话,可以点一下右下角的【推荐】。

JDK的命令行工具系列 (二) javap、jinfo、jmap的更多相关文章

  1. JDK的命令行工具系列 (三) jhat、jstack

    jhat: heapdump文件分析工具 在前两篇系列文章JDK的命令行工具系列 (一) jps.jstat.JDK的命令行工具系列 (二) javap.jinfo.jmap中, 我们已经介绍过了jp ...

  2. JDK的命令行工具系列 (一) jps、jstat

    概述 在我们进行故障定位和性能分析时, 可以使用Java Dump(也叫Dump文件)来帮助排查问题, 它记录了JVM运行期间的内存占用和线程执行等情况.其中Heap Dump文件是二进制格式, 它保 ...

  3. JDK的命令行工具

    Jcmd:综合工具 jcmd -l  列出当前运行的所有虚拟机 参数-l表示列出所有java虚拟机,针对每一个虚拟机,可以使用help命令列出该虚拟机支持的所有命令 jcmd [pid] help j ...

  4. 虚拟机性能监控与故障处理工具------JDK的命令行工具

    ①jps:虚拟机进程状况工具 功能:列出正在运行的虚拟机进程,并显示1.虚拟机执行主类名称以及2.这些进程的本地虚拟机唯一ID(LVMID). 使用频率最高的JDK命令行工具,其他的JDK工具大多需要 ...

  5. jdk(1.8)命令行工具(二)

    2.3 jinfo:java配置信息工具 jinfo(Configuration Info for Java)的作用是实时的查看和调整虚拟机的各项参数.使用jps -v 可以查看虚拟机启动时显示指定的 ...

  6. FFmpeg命令行工具学习(二):播放媒体文件的工具ffplay

    一.简述 ffplay是以FFmpeg框架为基础,外加渲染音视频的库libSDL构建的媒体文件播放器. 在使用ffplay之前必须要安装到系统中,MAC的安装教程为:http://www.cnblog ...

  7. jdk命令行工具系列

    虚拟机堆转储快照分析工具使用jmap等方法生成java的堆文件后jhat:虚拟机堆转储快照分析工具 导出程序执行的堆信息 jps jps -l jmap -dump:format=b,file=D:/ ...

  8. JDK常用命令行工具(基于JDK10)

    虽然我是在jdk10环境下, 但是大体上和jdk8是差不多的. 总共有这么多 本来想着一口气把所有命令都边学边总结一下的, 结果发现....有些还真的不是很常用....或者说我这个水平还接触不到那么多 ...

  9. JDK常用命令行工具使用

随机推荐

  1. 设计模式之策略模式和状态模式(strategy pattern & state pattern)

    本文来讲解一下两个结构比较相似的行为设计模式:策略模式和状态模式.两者单独的理解和学习都是比较直观简单的,但是实际使用的时候却并不好实践,算是易学难用的设计模式吧.这也是把两者放在一起介绍的原因,经过 ...

  2. J2SE的基本简介与J2EE/J2ME的差异

    J2SE简介与J2EE.J2ME的比较 Java2平台包括:标准版(J2SE).企业版(J2EE)和微缩版(J2ME)三个版本. J2SE,J2ME和J2EE,这也就是SunONE(Open NetE ...

  3. Docker搭建MySQL主从集群,基于GTID

    写在前边 搭建MySQL读写分离主从集群,这里未使用binlog方式,使用的是GTID方式 源码见我的Github https://github.com/hellxz/mysql-cluster-do ...

  4. CentOS7使用firewalld防火墙

    firewalld的基本使用 启动: systemctl start firewalld 关闭: systemctl stop firewalld 查看状态: systemctl status fir ...

  5. python爬虫的一个常见简单js反爬

    python爬虫的一个常见简单js反爬 我们在写爬虫是遇到最多的应该就是js反爬了,今天分享一个比较常见的js反爬,这个我已经在多个网站上见到过了. 我把js反爬分为参数由js加密生成和js生成coo ...

  6. 微服务-springboot-读写分离(多数据源切换)

    为什么需要读写分离 当项目越来越大和并发越来大的情况下,单个数据库服务器的压力肯定也是越来越大,最终演变成数据库成为性能的瓶颈,而且当数据越来越多时,查询也更加耗费时间,当然数据库数据过大时,可以采用 ...

  7. Golang之mirco框架部分浅析

    在实习中使用 micro 框架,但是挺多不懂的,看了部分源码搞懂了一些,还是有一些比较复杂没搞懂. 第一部分:初始化 service 并修改端口 main.go // waitgroup is a h ...

  8. Python之Pandas库学习(二):数据读写

    1. I/O API工具 读取函数 写入函数 read_csv to_csv read_excel to_excel read_hdf to_hdf read_sql to_sql read_json ...

  9. ZOJ 3795:Grouping(缩点+最长路)

    http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemId=5303 题意:有n个人m条边,每条边有一个u,v,代表u的年龄大于等于v,现在要 ...

  10. JDK源码阅读(三):ArraryList源码解析

    今天来看一下ArrayList的源码 目录 介绍 继承结构 属性 构造方法 add方法 remove方法 修改方法 获取元素 size()方法 isEmpty方法 clear方法 循环数组 1.介绍 ...