一、使用jdb调试的应用场景。

1、一般开发阶段调试bug可以使用开发工具(idea、eclipse)自带的调试插件进行本地调试和远程调试。使用开发工具自带的调试插件固然很方便、但是多掌握一门调试工具意味着能够多应对一种应用场景。

、但是当java程序处于受限环境下运行,比如应用服务器是隔离的内网环境,这时候在服务器上安装开发工具就显得那么繁琐。由于jdb是jdk自带的命令行调试工具,它本身是轻量级的,也省去了额外的安装环节。

二、jdb调试jar包

1、运行

java -Xdebug -Xrunjdwp:transport=dt_socket,server=y,suspend=n,address=8000 -jar test.jar archivessive-1.0.jar 命令启动被调试端

一般我们使用java -jar xxx.jar启动一个可执行的jar包,使用jdb进行调试,我们添加额外的命令行参数 -Xdebug -Xrunjdwp:transport=dt_socket,server=y,suspend=n,address=8000

该命令中dt_socket实际上对应一个dll动态链接库,通过查看jdk安装目录下的文件可以发现

java开发调试工具 java代码调试工具_java开发调试工具

通过dumpbin命令查看dt_socket.dll的导出函数发现jdwpTransport_OnLoad

java开发调试工具 java代码调试工具_java_02

 所以这个调试功能是使用native库实现的。

选项 suspend=y的时候 将会挂起主线程 

java开发调试工具 java代码调试工具_jar_03

选项suspend=n的时候将不会挂起主线程

java开发调试工具 java代码调试工具_java_04

 

 监听端口为8000,可以任意指定

jdb -connect com.sun.jdi.SocketAttach:hostname=localhost,port=8000 

启动调试端

java开发调试工具 java代码调试工具_开发工具_05

 以上图中是成功连接8000端口的示例

java开发调试工具 java代码调试工具_jar_06

以上是无法连接到8001端口 的示例

 使用 jdb成功连接被调试端后,就可以开始设置断点了

java开发调试工具 java代码调试工具_开发工具_07

上图中使用 methods 类的完全限定名称 命令就可以列举出这个类的所有方法 

java开发调试工具 java代码调试工具_java_08

上图中使用stop in 对类的searchByMultiCondition方法进行设置断点。 使用stop in 对方法设置断点的结果是 当断点命中的时候会断点停留在方法体的第一行,当然也可以使用stop at 针对行号进行设置断点

java开发调试工具 java代码调试工具_java开发调试工具_09

 

java开发调试工具 java代码调试工具_jar_10

此时运行对应的接口方法 将会命中断点,断点命中的时候 对应的线程将会挂起,此时可以在命令行输入相应的调试命令。比如可以查看本地变量的值、设置本地变量的值、转到下一步等等

java开发调试工具 java代码调试工具_jar_11

使用 locals命令可以列出相应的本地变量、此处方法的参数也是被作为本地变量来对待的 。

java开发调试工具 java代码调试工具_jar_12

 使用 eval 本地变量名 可以打印出变量的值、实际上是调用了变量的toString方法。

java开发调试工具 java代码调试工具_java开发调试工具_13

使用set命令可以修改本地变量的值、如上图id字段值被修改成了111 

java开发调试工具 java代码调试工具_jar_14

 

使用use java源代码路径  命令 可以结合源码进行调试,使用list命令可以看到当前断点在源码中的位置。

使用next 命令将会执行到下一行

java开发调试工具 java代码调试工具_jar_15

使用 run 命令执行到下一个断点位置

java开发调试工具 java代码调试工具_开发工具_16

为了不用每次重新设置断点和重复的命令,可以将预定义的命令放在ini文件中,

java开发调试工具 java代码调试工具_java开发调试工具_17

jdb.ini的内容为  

java开发调试工具 java代码调试工具_jar_18

将jdb调试端的命令放在批处理文件 中批量执行

java开发调试工具 java代码调试工具_java开发调试工具_19