在Thread.sleep(int)期间,JFrame处于非活动状态

时间:2015-09-08 13:35:34

标签: java multithreading swing events event-dispatch-thread

我有一些应该模拟旧计算机的Java类。其中一个是JFrame,名为 GUI ,带有 JTabbedPane 。计算过程需要一些时间,所以我已经包含了 Thread.sleep(int)。问题是,当计算功能正常工作时,我的GUI中的JTabbedPane的选项卡无法被选中。

以下是一些代码:

import java.util.*;
import java.awt.*;

public class Engine
{
    private GUI gui;
    private Store store;
    private Mill mill;

    public static void main(String[] args) {
        Engine e = new Engine();
    }

    public Engine() {
        gui = new GUI(this);
        mill = new Mill(this);
        store = new Store(this);

        store.draw();                       // affects GUI elements
    }

    public void read(String operations) {
        clearStatus();                      // affects GUI elements

        try {
            String[] op = operations.split("\n");

            for (int i = 0; i < op.length; i++) {
                setStatus(op[i]);        // affects GUI elements

                if (op[i].length() == 0) {  // empty line
                    continue;
                }

                int number = Integer.parseInt(op[i]);
                store.addNumber(number);

                Thread.sleep(200);
            }

            store.draw();                   // affects GUI elements

        } catch(Exception e) {
            System.out.println(e);
        }
    }

    public void clearStatus() {
        gui.statusLabel.setText("");
    }

    public void setStatus(String msg) {
        gui.statusLabel.setText(msg);
    }
}

这是我的GUI(这是一个简短的版本):

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

public class GUI extends JFrame
{
    private Engine engine;

    // Swing
    JTabbedPane tabs;
    JPanel mill;
    JTextArea input, store;
    JLabel statusLabel;
    JButton sendInput;

    public GUI(Engine e)
    {
        engine = e;

        input = new JTextArea();
        sendInput = new JButton("Run");
        sendInput.addActionListener(new ActionListener() {
            public void actionPerformed(ActionEvent evt) {
                pressInputButton(evt);
            }
        });

        JPanel inputPanel = new JPanel();
        inputPanel.add(input,     BorderLayout.CENTER);
        inputPanel.add(sendInput, BorderLayout.SOUTH);

        mill = new JPanel();
        statusLabel = new JLabel();
        mill.add(statusLabel);

        store = new JTextArea();

        JTextPane helpPane = new JTextPane();
        helpPane.setText("[...]");

        tabs = new JTabbedPane();
        tabs.addTab("Input", inputPanel);
        tabs.addTab("Mill", mill);
        tabs.addTab("Store", new JScrollPane(store));
        tabs.addTab("Help", helpPanel);

        add(tabs, BorderLayout.CENTER);
        setVisible(true);
        pack();
    }

    public void pressInputButton(ActionEvent evt) {
        engine.read(input.getText());
    }
}

1 个答案:

答案 0 :(得分:4)

@sorifiend的含义是,从事件处理程序调用sleep()将“锁定”应用程序的事件调度线程。在事件处理函数返回之前,应用程序不会响应任何其他用户输入。

如果您希望应用程序在计算器“正常工作”时做出响应,那么您需要做的是:

  • 记录计算器的状态为“正常工作”
  • 禁用用户在计算器处于“工作”状态时按下的任何按钮, - 或 - 如果按下这些按钮,则更改这些按钮的处理程序以执行不同的操作(模拟卡住机器?)处于“工作”状态。
  • 在不久的将来提交计时器事件一段时间(无论计算多长时间。)
  • 在计时器事件的处理程序中,显示计算结果,重新启用禁用按钮,并将状态从“工作”更改回“空闲”(或其他)。
相关问题