Intellij和eclipse的远程debug 均是基于JVM JPDA的调试
1. 远程debug使用方式:
工具上面 -> Edit Configuration... -> Remote -> new remote ->
这里需要填的有:
- Name: remote debug的名称,自定义
- Transport:Socket
- Debugger mode: Attach
- Host: 需要连接的远程主机名,必须填
- Port:监听的端口,必须填
-> Apply -> OK
在所想远程调试的服务器上,Tomcat的catalina.sh文件中,加入JAVA_OPTS变量
JAVA_OPTS = "-Xdebug -Xrunjdwp:transport=dt_socket,server=y,suspend=n,address=10086"
restart服务器,开启intellij或eclipse上的remote Debug即可
2. 远程debug的原理说明:
首先说明一下什么是JPDA:
JPDA体系结构主要包括:JVM TI,JDI, JDWP
JVM TI: JVM Tool Interface。 顾名思义是接口,这些接口支持调试并提供调试功能。接口的访问是以本地语言的形式提供的,由JVM负责实现。这些接口函数的直接客户端并不是调试器,而是有一个“JPDA back-end”。这个后端属于JVM的一部分,在SUN JRE的bin目录下可以找到jdwp.dll(jdwp.so)的库文件,这就是JPDA back-end的实现。JPDA back-end提供了各种访问方式(共享内存,Socket)接收调试器的请求,然后调用JVM TI接口。
JDI: Java Debug Interface。仍然是接口。处于JVM TI接口的上层API,以纯Java语言提供,由JDK实现(在Sun JDK的tools.jar可以找到)。调试器直接使用JDI来实现调试的功能。与JPDA back-end相对应,JDI实现的角色就是JPDA front-end。
JDWP:Java Debug Wire Protocol。定义了JPDA front-end和JPDA back-end之间通讯信息的格式协议。这里的通讯信息主要包括两种:调试器发送给JVM的请求信息和JVM发送给调试器的调试信息。
解释一下上面JAVA_OPTS
-Xdebug: 打开调试特性
-Xrunjdwp: 运行JDWP。包括:
- transport=dt_socket: 定义JDI和JVM TI 的通信格式,套接字传输
- server=y: y表示启动的JVM是被调试者。如果为n,则表示启动的JVM是调试器;
- address=10086:JVM监听的端口;
- suspend=n: n表示启动的JVM不会暂停等待,不直接将调试器连接上,需要手动启动。设置为y,则直接连接到调试器上
理清一下流程:Intellij 或Eclipse调试器调用JDK提供的JDI实现 (JPDA front-end),经过JDWP协议 ,和JPDA back-end 进行通讯。JPDA back-end 通过调用JVM TI接口 ,从而获知调试信息,或发送控制命令。然后,JPDA back-end 将调试信息或命令执行结果,通过JDWP协议 ,返回给调试器 。