一、使用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安装目录下的文件可以发现
通过dumpbin命令查看dt_socket.dll的导出函数发现jdwpTransport_OnLoad
选项 suspend=y的时候 将会挂起主线程
选项suspend=n的时候将不会挂起主线程
监听端口为8000,可以任意指定
2、运行 jdb -connect com.sun.jdi.SocketAttach:hostname=localhost,port=8000
启动调试端
以上图中是成功连接8000端口的示例
以上是无法连接到8001端口 的示例
使用 jdb成功连接被调试端后,就可以开始设置断点了
上图中使用 methods 类的完全限定名称 命令就可以列举出这个类的所有方法
上图中使用stop in 对类的searchByMultiCondition方法进行设置断点。 使用stop in 对方法设置断点的结果是 当断点命中的时候会断点停留在方法体的第一行,当然也可以使用stop at 针对行号进行设置断点
此时运行对应的接口方法 将会命中断点,断点命中的时候 对应的线程将会挂起,此时可以在命令行输入相应的调试命令。比如可以查看本地变量的值、设置本地变量的值、转到下一步等等
使用 locals命令可以列出相应的本地变量、此处方法的参数也是被作为本地变量来对待的 。
使用 eval 本地变量名 可以打印出变量的值、实际上是调用了变量的toString方法。
使用set命令可以修改本地变量的值、如上图id字段值被修改成了111
使用use java源代码路径 命令 可以结合源码进行调试,使用list命令可以看到当前断点在源码中的位置。
使用next 命令将会执行到下一行
使用 run 命令执行到下一个断点位置
为了不用每次重新设置断点和重复的命令,可以将预定义的命令放在ini文件中,
如
jdb.ini的内容为