4

我正在编写 aJToggleButton以从内存中加载/丢弃元素的配置(望远镜配置),因此我在 a 中添加了JComboBoxaJFrame并在其附近添加了按钮以加载所选项目。选择时JToggleButton,会显示一个硬盘图标,否则显示另一个图标。为此,我正在使用 IntelliJ IDEA GUI 编辑器。当然,我已经向该ItemListener按钮添加了一个(如网络建议):

    loadTelescopeButton.setSelected(true);
    System.out.println(loadTelescopeButton.isSelected());
    loadTelescopeButton.addItemListener(new ItemListener() {
        @Override
        public void itemStateChanged(ItemEvent e) {
            System.out.println("LAODACTION " + loadTelescopeButton.isSelected());
            try {
                if (e.getStateChange() == ItemEvent.SELECTED) {
                    String selected = telescopesList.getSelectedItem().toString();

                    if ((selected != null) && (!selected.equals("")) && (ObjUtils.isAlphaNumeric(selected))) {
                        //...

                    } else {
                        showErrorMessage("Invalid id selected!");
                    }

                } else if (e.getStateChange() == ItemEvent.DESELECTED) {
                    if ((configurationActivity != null) && (configurationActivity.getManager() != null) &&
                            (configurationActivity.getTelescope() != null) && (configurationActivity.getTelescope().isConnected())) {
                        //...

                    } else {
                        //...
                    }
                }

            } catch (Exception e1) {
                e1.printStackTrace();
            }
        }
    });

输出:
true
->显示窗口时
LAOD_ACTION false
->单击按钮时

我用一些新的切换按钮做了一些测试,它们给了我同样的错误:里面的代码itemStateChanged(ItemEvent e) {...}永远重复,没有停止!在那段代码中没有forwhile循环!结果是大量的消息对话框(应该只显示一个对话框),如果我在桌面上聚焦另一个窗口,对话框后面的屏幕变成黑色(父窗口的区域)。我将侦听器更改为ActionListener,现在一切都执行一次/单击。

为什么会出现这个错误?如您所见,我已从https://stackoverflow.com/a/7524627/6267019复制了该代码。

GitHub 上的完整代码 在这里,我突出显示了该切换按钮的代码。我的文件中的其他JToggleButtons也会发生相同的错误MainActivity.java,并且在调试 IntelliJ 时,我可以看到侦听器中的代码永远重复。经过数千次对话后,Windows 向我显示一条消息并关闭 Java Platform Binary 并出现错误。

编辑:
新课程中的同样问题:

 import javax.swing.*;
 import java.awt.*;

 public class ErrorGUI extends JFrame {

     public ErrorGUI() throws HeadlessException {
         super("ciao");
         JPanel panel1 = new JPanel();
         setContentPane(panel1);

         JToggleButton ciaoToggleButton = new JToggleButton("cajs");
         ciaoToggleButton.setSelected(true);
         ciaoToggleButton.addItemListener(e -> {
             System.out.println("caiooasfsdvn");
             try {
                 JOptionPane.showMessageDialog(panel1, "skjngksfnb");

             } catch (Exception e2) {
                 e2.printStackTrace();
             }
         });
         panel1.add(ciaoToggleButton);

         pack();
         setVisible(true);
     }

     public static void main(String[] args) {
         new ErrorGUI();
     }
 }
4

2 回答 2

4

每当您打开模态对话框时,打开方法调用只会在对话框关闭后返回。这对于返回输入值或选项的对话框至关重要。

这意味着当对话框打开时,必须启动一个新的事件处理循环来对对话框中的输入做出反应。

因此,当您从侦听器打开模态对话框时,您将停止处理当前事件并开始处理后续事件,这会严重干扰当前事件的处理。最值得注意的是,当打开新对话框时,按钮会突然失去焦点。

通过将监听器更改为

ciaoToggleButton.addItemListener(e -> {
    System.out.println("entering");
    JOptionPane.showMessageDialog(panel1,
       e.getStateChange()==ItemEvent.SELECTED? "selected": "deselected");
    System.out.println("leaving");
});

这将打印序列

entering
entering
leaving
leaving

显示在旧事件的处理尚未完成时如何生成矛盾事件。

正如其他人所说,您可以通过在完成均匀处理后打开对话框来解决此问题,例如

ciaoToggleButton.addItemListener(e -> {
    System.out.println("entering");
    EventQueue.invokeLater(() -> JOptionPane.showMessageDialog(panel1,
       e.getStateChange()==ItemEvent.SELECTED? "selected": "deselected"));
    System.out.println("leaving");
});

或者您强制执行非模态对话框:

ciaoToggleButton.addItemListener(e -> {
    System.out.println("entering");
    JDialog d = new JOptionPane(
            e.getStateChange()==ItemEvent.SELECTED? "selected": "deselected",
            JOptionPane.INFORMATION_MESSAGE)
        .createDialog(panel1, UIManager.getString("OptionPane.messageDialogTitle"));
    d.setModal(false);
    d.setVisible(true);
    System.out.println("leaving");
});

(在实际应用程序中,您可以保留对话框以供以后重用或dispose在使用后调用)


不幸的是,文档中没有足够强调打开模式对话框(或执行任何其他创建辅助事件循环的操作)的危险。您可以在任何地方读到从其他线程访问 Swing 组件可能会产生不一致,但是在有未完全处理的事件时启动新的事件处理循环可能会产生类似的影响。

于 2017-01-06T18:11:01.300 回答
3

我不能说我理解为什么您的代码行为不端,但我同意您所看到的不太有意义,并且可能是由于 JOptionPane 调用以某种方式影响了 JToggleButton 的状态更改。解决此问题的一种方法是将 JOptionPane 调用包装在 Runnable 中,并通过SwingUtilities.invokeLater(...). 例如:

import javax.swing.*;
import java.awt.*;

@SuppressWarnings("serial")
public class ErrorGUI extends JFrame {

    public ErrorGUI() throws HeadlessException {
        super("ciao");
        JPanel panel1 = new JPanel();
        setContentPane(panel1);

        JToggleButton ciaoToggleButton = new JToggleButton("cajs");
        ciaoToggleButton.setSelected(true);
        ciaoToggleButton.addItemListener(e -> {
            System.out.println("caiooasfsdvn");
            SwingUtilities.invokeLater(() -> {
                JOptionPane.showMessageDialog(panel1, "skjngksfnb");
            });
            // JOptionPane.showMessageDialog(panel1, "skjngksfnb");

        });
        panel1.add(ciaoToggleButton);

        pack();
        setDefaultCloseOperation(EXIT_ON_CLOSE);
        setLocationRelativeTo(null);
        setVisible(true);
    }

    public static void main(String[] args) {
        SwingUtilities.invokeLater(() -> {
            new ErrorGUI();
        });

    }
}

一个有趣的变化:

ciaoToggleButton.setSelected(true);
System.out.println("0:" + ciaoToggleButton.isSelected());
ciaoToggleButton.addItemListener(e -> {
    System.out.println("1: " + ciaoToggleButton.isSelected());
    if (e.getStateChange() == ItemEvent.SELECTED) {
        JOptionPane.showMessageDialog(panel1, "skjngksfnb");
    }
    System.out.println("2: " + ciaoToggleButton.isSelected());

});

打印出来:

0:true
1: false
2: false
1: true
1: false
2: false
2: false
1: true
1: false
2: false
2: false
于 2017-01-06T17:26:02.760 回答