如何强制 ImageJ 关闭所有 windows 而没有关闭事件错误?

How to force ImageJ to close all its windows without close event error?

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

异常

在 Linux 上使用 OpenJDK 7 时会发生这种情况。例外是 fixed in Java 8.

另请注意,该异常并不是您所看到的退出问题的真正原因。

处置问题

ImageJ 1.x 的应用程序处理一团糟。 (有关一些技术讨论,请参阅 this news post。)它实际上主要是 运行 作为一个独立的应用程序,并且主要在 exitWhenQuitting 标志设置为 true 的情况下进行测试,以便 JVM 关闭主 window 关闭后。因此,以不同的方式使用 ImageJ 导致图像挂起也就不足为奇了 windows.

我已经测试了各种解决方法——例如:

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();
        }
    }
});

但其中 none 的效果正如人们所希望的那样。根据上面链接发布的新闻,我花费了数周的时间进行开发和实验才能在 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 框架中的 "Close ImageJ" 按钮。您会看到 ImageJ 主 window 图像 window 根据需要进行处理(使用 this code from ImageJ Legacy)。

但是,存在(至少)三个问题:

  1. 此示例未将 ij.getContext().dispose() 调用连接到实际的 ImageJ1 UI window 关闭事件。这样做并非易事(我说最近没有深入研究这段代码)。

  2. 处理 ImageJ 以及额外的 JFrame 后,JVM 应该 关闭。实际上,我们为此付出了很多努力。但它实际上不适用于当前版本的 ImageJ,大概是由于某处有一些未处理的资源。这是一个错误。

  3. 单击主 ImageJ window 上的 X 将关闭整个 JVM,因为 ImageJ1 的 exitWhenQuitting 标志变为 set to true。您可以自己将其切换回 false,但这实际上很棘手,因为 class 加载问题与 ImageJ2 在 运行 时间使用 Javassist 修补 ImageJ1 相关。

下一个问题是:您真的有多需要它来工作?