JFrame:在语句之间休眠

JFrame: Sleep between statements

我对编程世界还是很陌生,最近注意到,每当我告诉程序空闲几秒钟代码之间,它就会在开始,然后完成剩余的代码。

我已经尝试了各种方法,例如 thread.sleep() 或定时器,但我从来没有得到我想要的东西。

这是一个例子:

public void Console(){

        Console.setDefaultCloseOperation(JFrame.DISPOSE_ON_CLOSE);
        Console.setSize(500, 500);
        Console.setLocationRelativeTo(null);
        Console.setResizable(false);
        Console.setVisible(true);
        Console.setTitle("Console");

        JPanel contentPane = new JPanel();
        contentPane.setBorder(new EmptyBorder(5, 5, 5, 5));

        Console.setContentPane(contentPane);
        contentPane.setLayout(null);
        contentPane.setBackground(new Color(47, 79, 79));

        cinput.setBounds(10, 442, 353, 20);
        contentPane.add(cinput);

        cinput.addActionListener(new ActionListener() {
            public void actionPerformed(ActionEvent e) {

                in();

                cinput.requestFocus();

            }
        });

        cinput.setColumns(10);
        cinput.requestFocus();

        JButton Enter = new JButton("Enter");
        Enter.setBounds(373, 439, 111, 23);
        contentPane.add(Enter);

        JScrollPane scrollPane = new JScrollPane();
        scrollPane.setBounds(10, 10, 474, 421);
        contentPane.add(scrollPane);

        cmd.setEditable(false);
        cmd.setFont(new Font("Courier New", Font.PLAIN, 18));
        cmd.setForeground(Color.GREEN);
        cmd.setText("CONSOLE\n");
        cmd.setBackground(Color.BLACK);
        cmd.setLineWrap(true);
        cmd.setWrapStyleWord(true);


        scrollPane.setViewportView(cmd);

        Enter.addActionListener(new ActionListener() {

            public void actionPerformed(ActionEvent arg0) {

            in();

            cinput.requestFocus();
            }
        });

    }

    private void addText(JTextArea textArea, String text) {

        String input = textArea.getText();
        String output = input + text;

        textArea.setText(output);
    }

    private void in()
    {
        String input = cinput.getText();
        cinput.setText("");

        String text;

        text = input;

        addText(cmd, "> " + text + "\n");

        if(text.equals("start"))
        {
            addText(cmd, "1");

            // SLEEP HERE

            Thread.sleep(1000); 

           // -------------------

            addText(cmd, "2");


        }
        else if(text.equals("exit"))
        {   
            Console.dispose();
        }
    }

它应该看起来像这样:

在这个非常基本的 'Console' 中,每当我在文本框中键入 'start' 并按回车键时,我希望数字“1”首先出现,1000 毫秒后数字“2”应该出现出现了,它没有!

有什么方法可以让程序在语句之间休眠而不是总是在函数开始时休眠?

提前致谢

在添加第一行文本之前添加睡眠语句:

Thread.sleep(1000);
addText(cmd, "1");

 // SLEEP HERE

Thread.sleep(1000); 

// -------------------

addText(cmd, "2");

看起来您正试图让主 Thread 休眠,这不是最好的主意。 运行 你自己的 Thread 并让它进入睡眠状态。

试试这个:

System.out.println(1);
SwingUtilities.invokeLater(() -> {
    new Timer(1000, new ActionListener() {
        @Override
        public void actionPerformed(ActionEvent e) {
            System.out.println(2);
        }
    }).start();
});

你也可以在没有 Thread.sleep() 的情况下这样做:

long tick = 1000;
long startTime = System.currentTimeMillis();

while (true) {
    long now = System.currentTimeMillis();
    while (now - startTime > tick) {
        System.out.println("Seconds from start time: " + (int) (tick / 1000));
        tick += 1000;
    }
}

如果它适合您,只需根据您的需要调整上面的示例(它还考虑了可能的延迟,因此秒数仍然是正确的)。

虽然从技术上讲上述答案是正确的,但您的问题是

  1. 您必须在 Swing 事件调度线程上完成所有 Swing 工作。
  2. 您真的、真的不想休眠主 Swing 事件调度线程!

您要做的是创建一个帮助程序 class 来包装 javax.swing.Timer(不是 java.util.timer),这样您就可以轻松地 运行 代码经过特定毫秒数后的事件线程。

public abstract class LazyAction implements Runnable { 

    public LazyAction(int millis) {
        Timer t = new Timer(millis, new ActionListener() {

                        @Override
                        public void actionPerformed(ActionEvent e) {
                         try{
                            LazyAction.this.run();
                         catch(Exception e){
                            //Your error handling code
                         }
                        }

                    });
                    t.setRepeats(false);
                    t.start();
       }}

现在,要使用此代码,您可以通过以下方式调用它:

  addText(cmd, "1");
  new LazyAction(300){

        @Override
        public void run() {
           addText(cmd, "2");
        }
  }

(请注意,如果您使用的是 Java 8,则可以改用 Lambdas,在这种情况下,LazyAction 将采用 Runnable 而不是实现 Runnable...)