JDI Java程序员的高级玩具
JDI,Java程序员的高级玩具
?
黄奕鹏(2013-08-30)?http://fly-hyp.iteye.com/
?
Word里面贴出来的格式不太正确,需要仔细看文章的网友,请看附件
?
JDI?即Java?Debug?Interface的简称,是JDK提供的调试接口,各种开发工具都是使用这个接口实现远程调试的。几个月前好好学习了一下,真的受益匪浅。JDI功能简单,接口简单,可以对开放调试端口的运行虚拟机做各种操作。直接使用JDI,与使用Eclipse?调试器比起来,它可以实现更有创意的功能。本文主要介绍一下JDI的基本使用,以及利用JDI可以完成的创意功能。
?
JDI的基本使用
1.?如何连接远程的Java虚拟机
?
1.1?为被调试的Java程序设置启动参数:
-Xrunjdwp:transport=dt_socket,address=8000,server=y,suspend=n
?
1.2??使用JDI连接远程虚拟机
使用Java?JDI?连接远程虚拟机,首先要得到VirtualMachine?对象
AttachingConnector?attacher=(AttachingConnector)?conn;?
VirtualMachine?vm=attacher.attach(arguments);?
不想贴大片的代码了,这里只想说连接成功以后就获得了VirtualMachine?对象。(详细代码请访问:http://www.ibm.com/developerworks/cn/java/j-lo-jpda4/index.html)
然后,通过?vm.?eventRequestManager()?得到EventRequestManager对象
?
2.?JDI?API的内部关系
?
JDI?API?是基于消息的,和Java?的?AWT?和?Swing的编程有点类似。主要是三个过程。
?
例如断点调试的过程
a.?注册一个特定断点的侦听事件。
EventRequestManager有下面这个方法
BreakpointRequest?createBreakpointRequest(Location?location)
?建立BreakpointRequest就相当于注册了这个位置断点的侦听事件
b.?远程虚拟机就会将相应的事件发送过来。
c.?轮询vm.eventQueue()就可以得到断点的事件。其中包含远程JVM当前执行位置的线程信息,局部变量等信息。通过这些运行上下文信息就可以加入各种的处理逻辑了。
?
3.?各种的Request以及处理关系图
EventRequestManager?可以各种创建Request。每一种Request代表一种特殊的调试的能力。?
4.?Request的各种Filter
Request包含各种Filter?由于限制消息的范围,消息多了必然会影响性能,通过各种Filter可以仅关注有用的消息。下面列出Filter的一些例子。
MethodEntryRequest?有下列Filter?
voidaddClassExclusionFilter(String?classPattern)
哪些类发生的事件,不被包含
voidaddClassFilter(ReferenceType?refType)
包含哪些类,通过接口或类的实例区分
voidaddClassFilter(String?classPattern)
包含哪些类
voidaddInstanceFilter(ObjectReference?instance)
实例过滤
voidaddThreadFilter(ThreadReference?thread)
线程过滤
?
因为Filter?非常灵活,各种Filter类型是不能通过界面型Java调试工具完全驾驭的。直接使用JDI将更加灵活和强大。
?
JDI?JavaDoc?地址:
http://docs.oracle.com/javase/7/docs/jdk/api/jpda/jdi/index.html
?
JDI可以实现的创意功能
1.?VSCM?以外调试代码
?
以下是一个方案可以解决Java在VSCM中调试VSCM外代码的问题??
调试工具在VSCM外面,可以不受VSCM限制的连接各个环境的虚拟机。在调试工具需要查看源代码时,再访问VSCM中的源代码服务程序,然后在VSCM中查看相关的源代码。
?
2.?跟踪方法的执行时间
?
通过直接使用JDI记录相关方法的执行时间,是没有侵入性的,而且简单。
例如
可以跟踪某个包下所有方法的运行时间
可以跟踪实现某个Interface的所有方法的运行时间
?
3.?添加方法的传入参数日志
?
一般使用调试工具,只能在方法入口处设置断点,然后查看传入参数的情况。这样必须长时间的中断线程的运行。
如果通过程序的方式,就可以几乎不中断线程执行,记录关注的方法的传入参数,记入日志。通过查看日志的方式对于某些开发问题的调试。
?
4.?动态代码替换
VirtualMachine类中有如下方法
?redefineClasses(Map<??extends?ReferenceType,byte[]>?classToBytes)
可以将用编译后的.class文件替换远程虚拟中的运行类。
这个是eclipse调试器中已有的功能,直接使用JDI也是挺简单的。
?
?
?
?
?
?
?
?
?
?
?
?
?
?
?
?
?