如何强制ImageJ关闭所有窗口而不会出现关闭事件错误?

时间:2015-11-05 11:50:59

标签: java imagej

我正在编写一个用于图像分析的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关闭其所有窗口的任何建议吗?

1 个答案:

答案 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)。

然而,(至少)有三个问题:

  1. 此示例未将ij.getContext().dispose()调用与实际的ImageJ1 UI窗口关闭事件挂钩。这样做不会是微不足道的(我说最近没有深入挖掘这段代码)。

  2. 在处理ImageJ以及额外的JFrame之后,JVM 假定关闭。实际上,我们付出了很多努力来实现这一目标。但它实际上并不适用于当前版本的ImageJ,可能是由于某些不相干的资源。这是一个错误。

  3. 单击主ImageJ窗口上的X会关闭整个JVM,因为ImageJ1的exitWhenQuitting标志会获得set to true。您可以自己将其切换回false,但由于ImageJ2在运行时使用Javassist修补ImageJ1这一事实导致的类加载问题,这实际上很棘手。

  4. 接下来的问题是:你真的需要这么多工作吗?