使用JDialog和Thread的ClassCastException

时间:2010-06-03 13:48:44

标签: java multithreading classcastexception jdialog

我有一个带有线程的JDialog。它在创建对话框时偶尔会给我一个ClassCastException(这意味着我可以成功地获得成功),而且我不知道应该在哪里发生。

这是我的JDialog类的片段

public class ConfirmExitDialog extends JDialog implements Runnable,
    ActionListener {
private static final long serialVersionUID = -8762051370686039110L;
private Thread dialogThread;
private boolean running;
private int result, count = 60;
private HandleExit  handleExit = null;

// GUI
private JOptionPane optionPane;
private JLabel msgLabel = new JLabel();
private JButton btnYes;
private JButton btnNo;

private void updateLabelText() {
    msgLabel.setText("<html>Ønsker du at afslutte dagens salg?<br>Programmet afslutter automatisk om " + count + " sekunder.</html>");
}

public int getResult() {
    return result;
}

public ConfirmExitDialog(Frame frame, HandleExit handleExit) {
    super(frame, false);
    this.handleExit = handleExit;

    setTitle("Afslut dagens salg?");
    display();

    running = true;
    if (dialogThread == null) {
        dialogThread = new Thread(this, "ConfirmExitDialog");
        dialogThread.start();
    }
    this.setModal(true);
}

public void close() {
    if (dialogThread != null)
        running = false;
}

private void display() {
    setLayout(new BorderLayout());

    // Buttons
    btnYes = new JButton("Ja");
    btnYes.addActionListener(this);
    btnYes.setMnemonic('J');
    add(btnYes, BorderLayout.WEST);
    btnNo = new JButton("Nej");
    btnNo.addActionListener(this);
    btnNo.setMnemonic('N');
    add(btnNo, BorderLayout.EAST);
    JButton[] buttons = { btnYes, btnNo };

    updateLabelText();
    optionPane = new JOptionPane(msgLabel, JOptionPane.QUESTION_MESSAGE,
            JOptionPane.YES_NO_OPTION, null, buttons, buttons[0]);
    setContentPane(optionPane);

    setDefaultCloseOperation(DISPOSE_ON_CLOSE);

    // Handle window closing correctly.
    setDefaultCloseOperation(DO_NOTHING_ON_CLOSE);
    addWindowListener(new WindowAdapter() {
      public void windowClosing(WindowEvent we) {
        /*
         * Instead of directly closing the window, we're going to change the
         * JOptionPane's value property.
         */
        optionPane.setValue(new Integer(JOptionPane.CLOSED_OPTION));
      }
    });

    pack();
    setVisible(true);   
}

private void countDown() {
    updateLabelText();
    count--;
}

@Override
public void run() {
    // TODO Auto-generated method stub

    try {
        display();
        while (running && count > 0) {
            System.out.println("Countdown " + count);
            pack();
            countDown();
            Thread.sleep(1000);
        }

        setVisible(false);
        if (count == 0)
            handleExit.closeApplication(true, true);


    } catch (InterruptedException ie) {
        // Thread stopped
    }
}

@Override
public void actionPerformed(ActionEvent event) {
    Object src = event.getSource();

    if (src == btnYes) {
        setVisible(false);
        result = JOptionPane.YES_OPTION;
        running = false;
        handleExit.closeApplication(true, false);
    }
    if (src == btnNo) {
        setVisible(false);
        result = JOptionPane.NO_OPTION;
        running = false;

    }
}

}

这是异常演员:

  

线程“AWT-EventQueue-0”中的异常   java.lang.ClassCastException at   javax.swing.LayoutComparator.compare(LayoutComparator.java:61)     在   java.util.Arrays.mergeSort(Arrays.java:1293)     在   java.util.Arrays.mergeSort(Arrays.java:1282)     在   java.util.Arrays.sort(Arrays.java:1210)     在   java.util.Collections.sort(Collections.java:159)     在   javax.swing.SortingFocusTraversalPolicy.enumerateAndSortCycle(SortingFocusTraversalPolicy.java:119)     在   javax.swing.SortingFocusTraversalPolicy.getFirstComponent(SortingFocusTraversalPolicy.java:434)     在   javax.swing.LayoutFocusTraversalPolicy.getFirstComponent(LayoutFocusTraversalPolicy.java:148)     在   javax.swing.SortingFocusTraversalPolicy.getDefaultComponent(SortingFocusTraversalPolicy.java:511)     在   java.awt.FocusTraversalPolicy.getInitialComponent(FocusTraversalPolicy.java:152)     在   java.awt.Window.getMostRecentFocusOwner(Window.java:2131)     在   java.awt.DefaultKeyboardFocusManager.dispatchEvent(DefaultKeyboardFocusManager.java:629)     在   java.awt.Component.dispatchEventImpl(Component.java:4502)     在   java.awt.Container.dispatchEventImpl(Container.java:2099)     在   java.awt.Window.dispatchEventImpl(Window.java:2478)     在   java.awt.Component.dispatchEvent(Component.java:4460)     在   java.awt.EventQueue.dispatchEvent(EventQueue.java:599)     在   java.awt.SequencedEvent.dispatch(SequencedEvent.java:101)     在   java.awt.EventQueue.dispatchEvent(EventQueue.java:597)     在   java.awt.EventDispatchThread.pumpOneEventForFilters(EventDispatchThread.java:269)     在   java.awt.EventDispatchThread.pumpEventsForFilter(EventDispatchThread.java:184)     在   java.awt.EventDispatchThread.pumpEventsForHierarchy(EventDispatchThread.java:174)     在   java.awt.EventDispatchThread.pumpEvents(EventDispatchThread.java:169)     在   java.awt.EventDispatchThread.pumpEvents(EventDispatchThread.java:161)     在   java.awt.EventDispatchThread.run(EventDispatchThread.java:122)

谢谢。 丹尼尔

3 个答案:

答案 0 :(得分:4)

强烈建议不要在Swing的EventDispatchThread以外的线程中进行修改,因为它可能会导致奇怪的副作用(就像你遇到的那样)。

想象一下,UI即将重新绘制自己(以及布局内容),您只需使用另一个线程同时更改UI的各个部分即可。在这种情况下,很可能会发生混乱。

有关详细信息,请参阅Java教程课程Concurrency in Swing

处理此问题的正确方法是让后台线程执行与UI无关的任何工作,并在EventDispatchThread中调度UI更新作业(将在处理事件之间执行作业)。

示例:

    Thread t = new Thread() {

        public void run () {

            // do background work

            SwingUtilities.invokeLater(new Runnable() {

                public void run () {
                    // update UI parts
                }
            });
        }
    };
    t.start();

SwingUtilities.invokeLater(Runnnable)将安排Runnable以供稍后执行,而 SwingUtilities.invokeAndWait(Runnnable)将安排一个Runnable并等待它被执行。

答案 1 :(得分:1)

Swing组件的所有更新都应该在Event Dispatch Thread(EDT)上完成。

在你的线程中,你每秒都尝试打包()对话框。此代码不在EDT上执行。

尝试使用SwingUtilities.invokeLater(...)来执行pack()。

或许应该使用SwingWorker而不是Thread。有关详细信息,请阅读Concurrency上的Swing教程中的部分。

或者甚至更好的方法是启动Swing Timer来安排关闭对话框。当Timer触发时,代码会自动在EDT上执行。 Swing教程还有一个关于使用Timers的部分。

答案 2 :(得分:0)

您正从Event Dispatcher线程以外的线程启动对话框。任何UI创建/更新都应该来自Swing的事件调度程序线程。

尝试:

SwingUtilities.invokeLater(new Runnable() {
    public void run() {
        // show the UI here (display method in your logic?)
    }
});

注意:在显示(可见)之前对UI的任何更新都可以从事件调度程序线程外部完成。显示用户界面后, 应始终在事件发送线程中进行任何更改。