我有自己的JDI,它对一些对象调用toString方法:
com.sun.jdi.ObjectReferenceobject object = ...
ThreadReference threadRef = frameProxy.threadProxy().getThreadReference();
Value value = object.invokeMethod(threadRef, toStringMethod,
Collections.EMPTY_LIST, ObjectReference.INVOKE_SINGLE_THREADED);问题是,即使在toString()方法中没有设置断点,invokeMethod也不会终止,所以我的调试器挂起。例如,当我在Double对象上调用它时,就会发生这种情况。
经过一段时间后,我如何才能杀死invokeMethod的执行呢?
更新:我尝试实现自己的Double对象,并在toString()方法的开头和结尾放置了一些System.out.println()语句,该方法似乎执行得很好,但出于某种原因,调试器没有收到结果。也许这是JDI中的一个bug,因为有很多这样的bug,但是我是,而不是,我只是在寻找一种方法来中止invokeMethod()的执行,如果需要太多的时间。
Update2:,我尝试了ThierryB的建议,但我只能在管理器线程中调用frameProxy.threadProxy().stop(object);。管理器线程因为invokeMethod()而被阻塞,所以它不会执行我的命令。我试过这样的方法:
boolean[] isFinished = new boolean[2];
isFinished[0] = false;
DebuggerManagerThreadImpl managerThread = debugProcess.getManagerThread();
new Thread(() - > {
try {
Thread.sleep(2000);
if (!isFinished[0]) {
System.out.println("Invoked");
managerThread.invokeCommand(new DebuggerCommand() {
@Override
public void action() {
try {
frameProxy.threadProxy().stop(object);
} catch (InvalidTypeException e) {
e.printStackTrace();
}
int threadStatus = frameProxy.threadProxy().status();
switch (threadStatus) {
case ThreadReference.THREAD_STATUS_RUNNING:
System.out.println("The thread is running.");
break;
case ThreadReference.THREAD_STATUS_ZOMBIE:
System.out.println("The thread has been completed.");
break;
case ThreadReference.THREAD_STATUS_WAIT:
System.out.println("The thread is waiting.");
break;
default:
System.out.println("The thread is not running / not waiting / not completed : but what is it doing right now ? (a little programmer joke ;) )");
}
}
@Override
public void commandCancelled() {
}
});
}
} catch (InterruptedException e) {
e.printStackTrace();
}
}).start();
Value value = object.invokeMethod(threadRef, toStringMethod,
Collections.EMPTY_LIST, ObjectReference.INVOKE_SINGLE_THREADED);
isFinished[0] = true;但是frameProxy.threadProxy().stop(object);从未被执行,因为DebuggerCommand's action方法没有被调用(线程被阻塞)。
此外,当我的调试器挂起并强制停止该过程时,这是一个堆栈跟踪:
com.sun.jdi.VMDisconnectedException
at com.jetbrains.jdi.TargetVM.waitForReply(TargetVM.java:317)
at com.jetbrains.jdi.VirtualMachineImpl.waitForTargetReply(VirtualMachineImpl.java:1170)
at com.jetbrains.jdi.PacketStream.waitForReply(PacketStream.java:86)
at com.jetbrains.jdi.JDWP$ObjectReference$InvokeMethod.waitForReply(JDWP.java:4840)
at com.jetbrains.jdi.ObjectReferenceImpl.invokeMethod(ObjectReferenceImpl.java:413)更新3:使用哪个线程调用该方法?目前,我使用的frameProxy.threadProxy().getThreadReference();大多数情况下都很好,但是是否应该创建一个单独的线程来调用对象上的方法(沿着我的JDI调试器,我在应用程序中也有一个检测代理,这样我就可以为这个用例创建一个单独的线程了(也许这样可以防止死锁)。
更新4:目前我使用SUSPEND_ALL作为挂起策略,使用SUSPEND_EVENT_THREAD更好吗?
发布于 2020-08-19 23:53:10
您可以将接口ThreadReference与方法void stop(ObjectReference throwable)一起使用。javadoc告诉“使用异步异常停止这个线程”。
try {
com.sun.jdi.ObjectReferenceobject object = ...
ThreadReference threadRef = frameProxy.threadProxy().getThreadReference();
frameProxy.threadProxy().stop(object);
int threadStatus = frameProxy.threadProxy().status();
switch(threadStatus) {
case ThreadReference.THREAD_STATUS_RUNNING :
log.info("The thread is running.");
break;
case ThreadReference.THREAD_STATUS_ZOMBIE :
log.info("The thread has been completed.");
break;
case ThreadReference.THREAD_STATUS_WAIT :
log.info("The thread is waiting.");
break;
default :
log.info("The thread is not running / not waiting / not completed : but what is it doing right now ? (a little programmer joke ;) )");
}
} catch (ClassNotLoadedException cnle) {
// log exception, or display a message...
} catch (IncompatibleThreadStateException itse) {
// log exception, or display a message...
} catch (InvalidTypeException ite) {
// log exception, or display a message...
}当您使用status方法检查线程状态时,可以找到以下值:
希望你能找到一种方法来做你需要的东西,我在这里提供。来自这里或下面的套件测试发表在阿尔文亚历山大的网站上的例子也会有所帮助。
https://stackoverflow.com/questions/63458738
复制相似问题