2

我有一个 Timer 用于扩展 JPanel 动画的类中,ActionListener 监听它并使 actionPerformed 运行,它会在需要时重新绘制并停止计时器。但是启动计时器的方法 animatePanel 在计时器运行时继续执行,这是我不想要的。我希望它等到计时器停止返回。

Timer 在类的构造函数中初始化,如下所示:

timer = new Timer(5, taskPerformer);

这就是它的作用。我有一个叫做 animatePanel() 的东西:

    private ActionListener taskPerformer = new ActionListener() {
    public void actionPerformed(ActionEvent evt) {
        ...
        if (some conditions){
            ...
            timer.stop();
            ...
            return;
        }

        ...
    }
};

private void animatePanel() {
    ...
    timer.start();
    System.out.println("Timer stopped."); //always executes before the timer has stopped :(
    //then returns and lets the rest of my program run while the timer is still going, which is BAD
}

计时器工作正常,除了在某些情况下, animatePanel() 会过早返回并让我的程序的其余部分运行,从而导致问题。

4

1 回答 1

2

您不能在 Event Dispatching Thread 的上下文中执行此操作,这样做会使您的应用程序挂起!

计时器必须在单独的Thread. 然后,您可以利用线程监控 API。

import java.awt.BorderLayout;
import java.awt.Dimension;
import java.awt.EventQueue;
import java.awt.Graphics;
import java.awt.Graphics2D;
import java.awt.GridBagConstraints;
import java.awt.GridBagLayout;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.util.logging.Level;
import java.util.logging.Logger;
import javax.swing.JButton;
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JPanel;
import javax.swing.Timer;
import javax.swing.UIManager;
import javax.swing.UnsupportedLookAndFeelException;

public class WaitForTimer {

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

    public WaitForTimer() {
        EventQueue.invokeLater(new Runnable() {
            @Override
            public void run() {
                try {
                    UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName());
                } catch (ClassNotFoundException | InstantiationException | IllegalAccessException | UnsupportedLookAndFeelException ex) {
                }

                JFrame frame = new JFrame("Testing");
                frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
                frame.setLayout(new BorderLayout());
                frame.add(new TestPane());
                frame.pack();
                frame.setLocationRelativeTo(null);
                frame.setVisible(true);
            }
        });
    }

    public static class TestPane extends JPanel {

        protected static final Object WAIT_FOR = new Object();
        private Timer timer;
        private int tickCount = 0;
        private JLabel ticks;
        private JButton start;

        public TestPane() {
            timer = new Timer(250, new ActionListener() {
                @Override
                public void actionPerformed(ActionEvent e) {
                    tickCount++;
                    if (tickCount > 10) {
                        tickCount = 0;
                        timer.stop();
                        synchronized (WAIT_FOR) {
                            WAIT_FOR.notifyAll();
                        }
                        start.setEnabled(true);
                    }
                    ticks.setText(String.valueOf(tickCount));
                }
            });
            timer.setRepeats(true);
            timer.setCoalesce(true);

            ticks = new JLabel("...");
            start = new JButton("Start");

            setLayout(new GridBagLayout());
            GridBagConstraints gbc = new GridBagConstraints();
            gbc.gridwidth = GridBagConstraints.REMAINDER;
            add(ticks, gbc);
            add(start, gbc);

            start.addActionListener(new ActionListener() {
                @Override
                public void actionPerformed(ActionEvent e) {
                    start.setEnabled(false);
                    new Thread(new Runnable() {
                        @Override
                        public void run() {
                            System.out.println("Starting timer...");
                            timer.start();
                            synchronized (WAIT_FOR) {
                                try {
                                    WAIT_FOR.wait();
                                } catch (InterruptedException ex) {
                                }
                            }
                            System.out.println("Timer finished...");
                        }
                    }).start();
                }
            });

        }

        @Override
        public Dimension getPreferredSize() {
            return new Dimension(200, 200);
        }
    }
}
于 2013-04-22T02:05:09.927 回答