我正在编写一个用于图像分析的Java应用程序,它一次打开ImageJ
和
ImageJ ij = new ImageJ();
并打开包含ImagePlus
的Windows。
现在,只要先关闭ImageJ
,按下关闭按钮时ImagePlus
将不关闭。另一种方式是工作,但在这两种情况下,在关闭ImageJ
之后抛出异常:
java.lang.reflect.InvocationTargetException
at java.awt.EventQueue.invokeAndWait(EventQueue.java:1288)
at java.awt.Window.doDispose(Window.java:1209)
at java.awt.Window.dispose(Window.java:1147)
at ij.ImageJ.run(ImageJ.java:784)
at java.lang.Thread.run(Thread.java:745)
Caused by: java.lang.IllegalArgumentException: null source
at java.util.EventObject.<init>(EventObject.java:56)
at java.awt.AWTEvent.<init>(AWTEvent.java:337)
at java.awt.event.InvocationEvent.<init>(InvocationEvent.java:285)
at java.awt.event.InvocationEvent.<init>(InvocationEvent.java:174)
at sun.awt.X11.XBaseMenuWindow.dispose(XBaseMenuWindow.java:907)
...
我不知道这两种情况是否相关。
有关如何强制ImageJ
关闭其所有窗口的任何建议吗?
答案 0 :(得分:2)
在Linux上使用OpenJDK 7时会发生这种情况。例外是fixed in Java 8。
另外:请注意,该异常不是您所看到的退出问题的实际原因。
ImageJ 1.x的应用程序处理是一个令人费解的混乱。 (有关技术讨论,请参阅this news post。)它主要用于作为独立应用程序运行,并且主要在exitWhenQuitting
标志设置为true的情况下进行测试,以便JVM在关闭时关闭主窗口。因此,以不同的方式使用ImageJ导致悬挂图像窗口就不足为奇了。
我测试了各种变通办法 - 例如:
ij.addWindowListener(new WindowAdapter() {
@Override
public void windowClosing(final WindowEvent e) {
// dispose all image windows
for (final int id : WindowManager.getIDList()) {
final ImagePlus imp = WindowManager.getImage(id);
if (imp == null) continue;
final ImageWindow win = imp.getWindow();
if (win != null) win.dispose();
}
// dispose all other ImageJ windows
for (final Window w : WindowManager.getAllNonImageWindows()) {
w.dispose();
}
}
});
但他们都没有像人们希望的那样工作。根据上面发布的新闻,我需要花费数周时间进行开发和实验,以便在ImageJ2中按照我们的要求进行戒烟工作。
以下是使用ImageJ2的一些代码,几乎的行为符合您的要求:
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import javax.swing.JButton;
import javax.swing.JFrame;
import javax.swing.WindowConstants;
import net.imagej.ImageJ;
public class IJDispose {
public static void main(final String... args) {
final ImageJ ij = new ImageJ();
ij.ui().showUI();
final JFrame frame = new JFrame("Hello");
final JButton b = new JButton("Close ImageJ");
b.addActionListener(new ActionListener() {
@Override
public void actionPerformed(final ActionEvent e) {
ij.getContext().dispose();
}
});
frame.getContentPane().add(b);
frame.setDefaultCloseOperation(WindowConstants.DISPOSE_ON_CLOSE);
frame.pack();
frame.setVisible(true);
}
}
启动后,按Shift + B打开Blobs样本图像。然后单击非ImageJ帧中的“关闭ImageJ”按钮。您将看到ImageJ主窗口和图像窗口根据需要进行处理(使用this code from ImageJ Legacy)。
然而,(至少)有三个问题:
此示例未将ij.getContext().dispose()
调用与实际的ImageJ1 UI窗口关闭事件挂钩。这样做不会是微不足道的(我说最近没有深入挖掘这段代码)。
在处理ImageJ以及额外的JFrame
之后,JVM 假定关闭。实际上,我们付出了很多努力来实现这一目标。但它实际上并不适用于当前版本的ImageJ,可能是由于某些不相干的资源。这是一个错误。
单击主ImageJ窗口上的X会关闭整个JVM,因为ImageJ1的exitWhenQuitting
标志会获得set to true。您可以自己将其切换回false,但由于ImageJ2在运行时使用Javassist修补ImageJ1这一事实导致的类加载问题,这实际上很棘手。
接下来的问题是:你真的需要这么多工作吗?