# JDK工具
# Flag
- https://docs.oracle.com/javase/8/docs/technotes/tools (opens new window)
- http://openjdk.java.net/projects/code-tools (opens new window)
- https://github.com/openjdk/jmc (opens new window)
- https://github.com/topics/diagnosis (opens new window)
- https://github.com/topics/trace (opens new window)
- https://github.com/topics/agent (opens new window)
其他命令
javac -encoding utf-8 -cp ".;webapp\WEB-INF\lib\*" -d target src\com\bajins\utils\*.java
java -cf bajins.war .
# 获取当前JVM默认参数
java -XX:+PrintFlagsFinal -version | grep MaxHeapSize
# 找出占用CPU高(或执行时间长)的线程PID
top -H -p $(pgrep java)
top -Hp $(pgrep java)
# TID等同PID
ps -mp $(pgrep java) -o THREAD,tid,time | sort -rn
# 打印堆栈异常信息,过滤转换为16进制的PID,-A 30显示后30行
jstack PID | grep $(printf "%x\n" PID) -A 30
jstack PID | grep -A 10 $(printf "%x\n" PID)
# nid:对应的linux操作系统下的TID,就是前面转化的16进制数字
# tid:这个应该是jvm的jmm内存规范中的唯一地址定位
jmap -dump:live,format=b,file=/tmp/heapdump.hprof PID
/usr/lib/jvm/jdk-YOUR-VERSION/bin/jcmd PID GC.heap_dump /tmp/heapdump.hprof
远程监控参数
在jvm启动参数中加入或在Tomcat的
/bin/catalina.sh
文件中加入
-Djava.rmi.server.hostname=主机的IP
-Dcom.sun.management.jmxremote.port=端口号
-Dcom.sun.management.jmxremote.ssl=false
-Dcom.sun.management.jmxremote.authenticate=false
远程Debug参数
附加调试器到进程:注意参数一定要放在
-jar
命令之前,方可运行成功
java -Djavax.net.debug=all -Xdebug -Xnoagent -Djava.compiler=NONE \
-Xrunjdwp:transport=dt_socket,server=y,suspend=n,address=端口 -jar jar包
-agentlib:jdwp=transport=dt_socket,server=y,suspend=n,address=端口 -jar jar包
参数说明
-Djavax.net.debug
查看调试信息,all
代表所有,其他有SSL
,handshake
,date
,trust manager
-Xdebug
是通知JVM工作在DEBUG模式下-Xnoagent
禁用默认sun.tools.debug调试器。-Djava.compiler=NONE
为了加快debug的速度,禁止JIT编译器的加载,详细说明 (opens new window)-Xrunjdwp
是通知JVM使用(Java debug wire protocol)来运行调试环境。transport=dt_socket
是指调试数据用SOCKET模式传送transport=dt_shmem
指用共享内存方式transport=dt_shmem
只适用于Windows平台。server=y/n
VM是否需要作为调试服务器执行。suspend=y/n
是否在调试客户端建立连接之后启动 VM 。(设置为y时启动不了)onthrow=java.io.IOException
指明,当产生该类型的Exception时,JVM就会中断下来,进行调式。可选参数launch=/sbin/echo
指明,当JVM被中断下来时,执行的可执行程序。可选参数onuncaught=y/n
指明,出现uncaught exception 后,是否中断JVM的执行.
# 监控分析
- Java诊断利器Arthas https://github.com/alibaba/arthas (opens new window)
- https://github.com/vipshop/vjtools (opens new window)
- https://github.com/qunarcorp/bistoury (opens new window)
- https://github.com/oldmanpushcart/greys-anatomy (opens new window)
- https://github.com/jvm-profiling-tools (opens new window)
- 无侵入式的jvm监控 https://github.com/ThinkpadNC5/MyPerf4J (opens new window)
- Java和Android堆转储分析器 https://heaphero.io (opens new window)
- GC 日志分析器 https://gceasy.io (opens new window)
- https://github.com/microsoft/gctoolkit (opens new window)
- https://github.com/chewiebug/GCViewer (opens new window)
- java程序跟踪工具 https://github.com/btraceio/btrace (opens new window)
- 故障排除,监视和性能分析 https://github.com/aragozin/jvm-tools (opens new window)
- https://github.com/ajermakovics/jvm-mon (opens new window)
- https://github.com/saleson/fm-dynamic-compiler (opens new window)
- https://github.com/gperftools/gperftools (opens new window)
- https://www.yourkit.com/java/profiler (opens new window)
- 内存分析 https://projects.eclipse.org/projects/tools.mat (opens new window)
- 监控统计 https://github.com/worstcase/gumshoe (opens new window)
- https://github.com/stevensouza/automon (opens new window)
- https://github.com/zrbcool/pepper-metrics (opens new window)
- 流分析 https://github.com/wavefrontHQ/wavefront-proxy (opens new window)
- JProfiler https://www.ej-technologies.com (opens new window)
# 反编译工具
java instrumentation
Java探针技术- https://github.com/topics/reverse-engineering (opens new window)
- https://github.com/topics/decompiler (opens new window)
- https://github.com/topics/disassembler (opens new window)
- 在线反编译器 http://www.javadecompilers.com (opens new window)
- https://github.com/java-decompiler (opens new window)
- https://github.com/leibnitz27/cfr (opens new window)
- https://github.com/mstrobel/procyon (opens new window)
- Gui for Procyon https://github.com/deathmarine/Luyten (opens new window)
- https://github.com/fesh0r/fernflower (opens new window)
- https://github.com/MaxPixelStudios/MinecraftDecompiler (opens new window)
- Jar和APK逆向工程套件 https://github.com/Konloch/bytecode-viewer (opens new window)
- dex2jar https://github.com/pxb1988/dex2jar (opens new window)
- https://github.com/horsicq/DIE-engine (opens new window)
- http://www.gda.wiki:9090/index.php (opens new window)
- https://ghidra-sre.org (opens new window)
- https://github.com/skylot/jadx (opens new window)
- https://github.com/kwart/jd-cmd (opens new window)
- https://software.intel.com/articles/pin-a-dynamic-binary-instrumentation-tool (opens new window)
- https://github.com/Col-E/Recaf (opens new window)
- smali/baksmali https://github.com/JesusFreke/smali (opens new window)
- snowman https://derevenets.com (opens new window)
javap -verbose 文件
查看class文件内容 https://github.com/topics/bytecode (opens new window)- 字节码查看器 https://github.com/ingokegel/jclasslib (opens new window)
- https://github.com/ClassViewer (opens new window)
- https://github.com/zxh0/classpy (opens new window)
- 反混淆 https://github.com/java-deobfuscator/deobfuscator (opens new window)
- jlink - 将一组模块及其依赖项组装并优化为自定义运行时映像 (opens new window)
- JDK14之jpackage命令 (opens new window)
- 大家都是怎么发布Java客户端程序的?难道让用户自己装JRE? (opens new window)
# tools
细心的可能会发现,
jdk/bin
目录下的这些工具都非常小, 是因为这些工具大多是jdk/lib/tools.jar
类库的一层包装而已,他们主要的功能代码是在tools
类库中实现的。
# 基本工具
这些工具是JDK的基础,用这些工具来编写应用程序。
工具名称 | 用途 |
---|---|
javac.exe | Java语言编译器 |
java.exe | Java应用程序启动器 |
javaw.exe | Java运行工具,用于运行.class字节码文件或.jar文件,但不会显示控制台输出信息,适用于运行图形化程序。 |
javadoc.exe | Java API文档生成器 |
apt.exe | java 注释处理器 |
appletviewer.exe | java applet小程序查看器 |
jar.exe | java文件压缩打包工具 |
jdb.exe | Java调试器 |
javah.exe | C头文件和stub生成器,用于写本地化方法,例如生产JNI样式的头文件 |
javap.exe | class文件反编译工具 |
extcheck.exe | 用于检测jar包中的问题 |
jcmd.exe | Java命令行(Java Command),用于向正在运行的JVM发送诊断命令请求。 |
# 安全工具
这些工具用于设置系统的安全规则和生产可以工作在远端的安全规则下的应用程序
工具名称 | 用途 |
---|---|
keytool.exe | 管理密钥库和证书 |
jarsigner.exe | 生产和校验JAR签名 |
policytool.exe | 有用户界面的规则管理工具 |
kinit.exe | 用于获得和缓存网络认证协议Kerberos 票证的授予票证 |
klist.exe | 凭据高速缓存和密钥表中的 Kerberos 显示条目 |
ktab.exe | 密钥和证书管理工具 |
# Java国际化工具
这些工具可以帮助你创建可本地化的应用程序
工具名称 | 用途 |
---|---|
native2ascii.exe | 将文本转化为 Unicode Latin-1 |
# 远程方法调用工具
这些工具可以帮助创建可以和web和网络交互的应用程序
工具名称 | 用途 |
---|---|
rmic.exe | 生成远程对象的stubs and skeletons(存根和框架) |
rmid.exe | Java远程方法调用(RMI:Remote Method Invocation)活化系统守护进程 |
rmiregistry.exe | Java远程对象注册表 |
serialver.exe | 返回类的 serialVersionUID |
java-rmi.exe | Java远程方法调用(Java Remote Method Invocation)工具,主要用于在客户机上调用远程服务器上的对象 |
# IDL和RMI-IIOP工具
这些工具用于创建使用OMG-Standard IDL 和 CORBA/IIOP 的应用程序
工具名称 | 用途 |
---|---|
tnameserv.exe | Java IDL瞬时命名服 |
idlj.exe | 生产映射到OMG IDL接口可以使Java应用程序使用CORBA的.java文件 |
orbd.exe | 为客户可以在CORBA环境下透明的定位和调用服务器的稳定的对象提供支持 |
servertool.exe | 为应用程序提供易于使用的接口用于注册,注销,启动,关闭服务器 |
# Java部署工具
工具名称 | 用途 |
---|---|
pack200.exe | 使用java gzip压缩工具将JAR文件转换为压缩的pack200文件,生产打包文件是高度压缩的JAR包,可以直接部署,减少下载时间 |
unpack200.exe | 解包pack200文件为JARs |
# Java web工具
工具名称 | 用途 |
---|---|
javaws.exe | Java web 启动命令行工具 |
schemagen.exe | Java构架的XML Schema生成器 |
wsgen.exe | 生成 JAX-WS |
wsimport.exe | 生成 JAX-WS |
xjc.exe | 绑定编译器 |
# JDK监控工具
Java故障检修,程序概要分析,监视和管理工具
工具名称 | 用途 |
---|---|
jvisualvm.exe | 一个图形化的Java虚拟机。从OracleJDK9开始,不再包含。 |
jconsole.exe | java监视台和管理控制台 |
jps.exe | JVM Process Status进程状态工具。列出目标系统的HotSpot JVM |
jstat.exe | 按照命令行的具体要求记录和收集一个JVM的性能数据 |
jstatd.exe | JVM jstat 的守护进程 |
jmc.exe | Java任务控制工具(Java Mission Control),主要用于HotSpot JVM的生产时间监测、分析、诊断。从OracleJDK9开始,不再包含。 |
# VisualVM
从
Oracle JDK 9
开始,不再包含。需到GitHub下载https://github.com/oracle/visualvm/releases (opens new window)
Oracle JDK 8
及以下版本在/jdk/bin
目录下jvisualvm.exe
需要注意的是:当OS所在分区是FAT格式时,VisualVM无法获取相关信息!
- 显示虚拟机进程以及进程的
配置
、环境信息
(jps
、jinfo
)。 - 监视应用程序的
CPU
、GC
、堆
、方法区(1.7及以前),元空间
(JDK1.8及以后)以及线程
的信息(jstat
、jstack
)。 - dump以及分析
堆转储快照
(jmap
、jhat
)。 - 方法级的
程序运行性能分析
,找出被调用最多、运行时间最长的方法。 - 离线程序快照:收集程序的运行时配置、线程dump、内存dump等信息建立一个快照
参考:基于JVisualVM的可视化监控 (opens new window), VisualVM监控远程阿里云主机 (opens new window)
不受此JVM支持
解决
Visual GC
提示不受此JVM支持
,要监控的主机没有配置jstatd
参考:jvisualvm 连接 jstatd 远程监控 jvm 或 Visual GC提示"不受此JVM支持“ (opens new window)
- 先查看
jstatd
服务是否可用
jps -l 127.0.0.1
- 一、在原有配置文件
java.policy
中添加
vi $JAVA_HOME/jre/lib/security/java.policy
- 在文件末位的
};
前添加
permission java.security.AllPermission;
- 启动jstatd
cd $JAVA_HOME/bin
./jstatd -J-Djava.security.policy=all.policy -J-Djava.rmi.server.hostname=本服务器IP -p 端口 &
- 查看运行端口情况
netstat -anp | grep jstatd
netstat -ntlp
lsof -i:1099
- 二、(推荐)新建一个配置文件
jstatd.all.policy
cd $JAVA_HOME/bin/
vi jstatd.all.policy
- 添加以下代码
grant codebase "file:${java.home}/../lib/tools.jar" {
permission java.security.AllPermission;
};
- 给文件加上执行权限
chmod +x jstatd.all.policy
- 在Java的bin目录下用以下命令启动
./jstatd -J-Djava.security.policy=jstatd.all.policy -J-Djava.rmi.server.hostname=本服务器IP -p 端口 -J-Djava.rmi.server.logCalls=true &
-J-Djava.security.policy=jstatd.all.policy
文件的绝对路径;-J-Djava.rmi.server.logCalls=true
打开日志,如果客户端有连接过来的请求,可以监控到,便于排错;-J-Djava.rmi.server.hostname=本服务器IP
指明本机 hostname 对应的本机地址,确保该地址可以给客户机访问。- 因为有的服务器 hostname 对应的 ip 不一定是外网能连上的,最好在这里直接明确指定;
-p 3333
指定服务的端口号,默认是1099。也是可选参数。
# JConsole
在
/jdk/bin
目录下jconsole.exe
,如果上面的内存
页签相当于可视化的jstat
命令的话,线程
页签的功能相当于可视化的jstack
命令,遇到线程停顿时可以使用这个页签进行监控分析。
线程长时间停顿的主要原因主要有:等待外部资源(数据库连接、网络资源、设备资源等)、死循环、锁等待(活锁和死锁)
查看hostname
hostname -i
如果hostname为
127.0.0.1
就需要修改,vi /etc/hosts
将其第一行127.0.0.1 localhost.localdomain localhost
中的127.0.0.1
修改为:本服务器IP
,重启Linux,在服务器上输入hostname -i
,查看实际设置的IP地址是否为你设置的
# jps
jps
命令(帮助文档 (opens new window)) 用于列出正在运行的虚拟机进程信息,
命令格式如下:
jps [ -q ] [ -mlvV ][hostid ]
jps [ -help ]
在默认情况下,jps的输出信息包括 Java 进程的进程ID以及主类名。jps还提供一些参数用于打印详细的信息。
其中-q
仅显示虚拟机的进程id,-mlvV
的意义如下:
-m
将打印传递给主类的参数-l
将打印模块名以及包名-v
将打印传递给虚拟机的参数-V
将打印传递给主类的参数、jar文件名等
# jstat
jstat
(帮助文档 (opens new window)) 是用于监视虚拟机各种运行状态信息的命令行工具,它可以显示本地或者远程虚拟机进程中的类加载、内存、垃圾回收等信息
命令格式如下
用法:jstat -help|-options
jstat -<option> [-t] [-h<lines>] <vmid> [<interval> [<count>]]
定义:
<option> -options选项报告的选项
<vmid>虚拟机标识符。 vmid采用以下形式:
<lvmid> [@ <主机名> [:<端口>]]
其中<lvmid>是目标的本地vm标识符
Java虚拟机,通常是进程ID; <hostname>是
运行目标Java虚拟机的主机的名称;
和<port>是rmiregistry的端口号
目标主持人。有关更完整的信息,请参阅jvmstat文档
虚拟机标识符的描述。
<lines>标题行之间的样本数。
<interval>采样间隔。允许以下表格:
<N> [ “MS” | “S”]
其中<n>是一个整数,后缀指定单位为
毫秒(“ms”)或秒(“s”)。默认单位为“ms”。
<count>终止前要采取的样本数。
-J <flag>将<flag>直接传递给运行时系统。
其中
vmid
全称是Virtual Machine Identifier
,就是jps命令显示的进程id,如果是远程虚拟机进程
vmid
的格式如下
[protocol:][//]lvmid[@hostname[:port]/servername]
jstat
命令包含很多的子命令,主要分为3类
- 类加载(
-class
) - 即时编译(
-compiler
和-printcompilation
) - 垃圾回收(
-gc*
)
输入jstat -options
显示如下
-class
-compiler
-gc
-gccapacity
-gccause
-gcmetacapacity
-gcnew
-gcnewcapacity
-gcold
-gcoldcapacity
-gcutil
-printcompilation
jstat [ generalOption | outputOptions vmid [interval[s|ms] [count]] ]
generalOption
一般使用-gcutil查看GC情况vmid
虚拟机进程号,即当前运行的java进程号interval
间隔时间,单位为秒或毫秒count
打印次数,如果缺省则打印无数次
# 每2秒输出一次内存情况,连续输出100次
jstat -gcutil <pid> 2000 100
jstat -gcutil $(pgrep java) 2000 100
# 输出heap各个分区大小,垃圾收集情况
# 每隔2秒打印一次,共打印2次
jstat -gc <pid> 2s 2
jstat -gc $(pgrep java) 2s 2
JVM堆是分代的,前四个表示
Survivor
区的容量(Capacity)和已使用量(Utilization),EC表示当前Eden的容量,在翻阅文档的时候, 发现没有CGC
和CGCT
的解释,它们分别代表并发GC Stop-The-World
的次数和时间。
# 故障检测和修理工具
工具名称 | 用途 |
---|---|
jinfo.exe | 配置或打印某个Java进程VM flag |
jhat.exe | 堆储存查看器 |
jmap.exe | Java内存图 |
jsadebugd.exe | Java的 Serviceability Agent Debug的守护进程 |
jstack.exe | Java堆栈跟踪 |
# jinfo
jinfo
命令(帮助文档 (opens new window)) 用来实时地查看和调整虚拟机的各项参数。
我们可以使用
jps -v
来查看传递给虚拟机的参数,即System.getProperty
获取的-D
参数,现在我们可以利用jinfo
命令来获取了。
命令格式如下
jinfo [option] pid
-flag name
打印指定的虚拟机参数的名称和值-flag [+|-]name
用来修改目标Java
进程的manageable
虚拟机参数。其中+
代表开启,-
代表关闭。命令:java -XX:+PrintFlagsFinal -version | grep manageable
-flag name=value
设置指定的虚拟机参数的值-flags
打印全部的虚拟机参数,例如:jinfo -flags 26792
-sysprops
打印java
系统参数(Java System Properties
)
# jmap
jmap
命令(帮助文档 (opens new window)) 用于生成堆转储快照,用于分析Java虚拟机堆中的对象。
命令格式为
jmap [options] pid
参数选项
-clstats
连接到正在运行的进程并打印Java堆被加载类的统计信息-finalizerinfo
连接到正在运行的进程并打印所有待 finalize 的对象。-histo[:live]
连接到正在运行的进程并统计各个类的实例数目以及占用内存,并按照内存使用量从多至少的顺序排列。只统计堆中还在存活的对象。-dump
连接到正在运行的进程并导出Java虚拟机堆内存的快照。live
只保存堆中存活的对象format=b
将使jmap
导出与hprof
(在Java 9
中已被移除)-XX:+HeapDumpAfterFullGC
、-XX:+HeapDumpOnOutOfMemoryError
格式一样的文件file=filename
指定导出堆内存快照的位置
# 生成dump文件
jmap -dump:live,format=b,file=heap.hprof $(pgrep java)
# jstack
jstack
命令(帮助文档 (opens new window)) 可以用来打印目标Java
进程中各个线程的栈轨迹,以及这些线程所持有的锁。
通过线程的栈轨迹可以定位线程长时间停顿的原因,如线程间死锁、死循环、请求外部资源导致长时间等待等。
命令格式如下
用法:
jstack [-l] <pid>
(连接到正在运行的进程)
jstack -F [-m] [-l] <pid>
(连接到挂起的进程)
jstack [-m] [-l] <executable> <core>
(连接到核心文件)
jstack [-m] [-l] [server_id @] <远程服务器IP或主机名>
(连接到远程调试服务器)
选项:
-F强制进行线程转储。 当jstack <pid>没有响应时(进程挂起)时使用
-m打印java和本机帧(混合模式)
-l长列表。 打印有关锁的其他信息
-h或-help打印此帮助消息
线程的状态
在输出的信息中,会包含,下面是常见的线程状态:
RUNNABLE
线程处于执行中BLOCKED
线程被阻塞WAITING
线程正在等待TIMED_WAITING
超时等待
示例
# 获取到线程的dump日志
jstack -l $(pgrep java) >dump.log
# 将所有线程信息输入到指定文件中
jstack -F $(pgrep java) > jvm.log
# jcmd
jcmd
命令(帮助文档 (opens new window)) 可以向运行中的Java
虚拟机(JVM
)发送诊断命令。
命令格式如下
jcmd <pid | main class> <command ... | PerfCounter.print | -f file>
jcmd -l
jcmd -h
参数
pid
虚拟机的进程idmain class
接收诊断命令请求的进程的main类。command
该命令必须是针对所选JVM的有效
jcmd
命令。
jcmd
的可用命令列表是通过运行help
命令(jcmd pid help
)获得的,其中pid
是运行Java
进程的进程ID。
如果
pid
为0,命令将被发送到所有的Java
进程。main class
参数将用于部分或完全匹配用于启动Java
的类。
如果没有提供任何选项,会列出正在运行的
Java
进程标识符以及用于启动进程的主类和命令行参数(与使用-l
相同)。
Perfcounter.print
打印目标Java
进程上可用的性能计数器。性能计数器的列表可能会随着Java
进程的不同而产生变化。-f file
从文件file中读取命令,然后在目标Java
进程上调用这些命令。-l
查看所有的进程列表信息。-h
查看帮助信息。(同-help
)
查看可用命令
jcmd $(pgrep java) help
示例
# 生成dump文件
jcmd <pid> GC.heap_dump /home/heap.hprof
# 其他工具
工具名称 | 用途 |
---|---|
jabswitch.exe | Java Access Bridge Switch的简称,用于控制Java访问桥的开/关。Java访问桥是一种技术,让Java应用程序实现Accessibility API,以供Microsoft Windows系统的辅助技术访问。 |
javafxpackager.exe | JavaFX打包工具 |
jrunscript.exe | 运行脚本 |