使用Thread.sleep()的paintComponent(Graphics g)的时间延迟不能按预期工作

时间:2012-11-16 09:56:24

标签: java swing progress-bar paintcomponent event-dispatch-thread

我正在创建一个动画ProgressBar,其中我使用了多个fillRect()javax.swing.Graphics方法。
为了在绘制每个矩形后延迟,我使用Thread.sleep(500)方法进行延迟,(由许多论坛推荐,为了延迟)。
问题是,在显示每个矩形框后,不是延迟0.5秒,而是在开始时占用所有矩形所需的整个延迟,然后显示最终图像,即进度条。
问题1
为了延迟每一个酒吧,我把延迟" Thread.sleep(500)"以及酒吧" fillRect()"在一个for() loop中,我想知道,为什么它在开始时需要所有延迟,然后才会显示已完成的ProgressBar。
问题2
如何更改我的代码,以便延迟可以与每个矩形条同时发生,因此当我运行程序时,它应该生成一个动画进度条。
代码:

import javax.swing.JOptionPane;
import java.awt.Graphics;
import javax.swing.JFrame;
import javax.swing.JPanel;
import java.awt.Color;

class DrawPanel extends JPanel
{
    public paintComponent(Graphics g)
    {
        super.paintComponent(g);
        g.setColor(new Color(71,12,3));
        g.fillRect(35,30,410,90);

        for ( int i=1; i<=40; i+=2)
        {
          Color c = new Color (12*i/2,8*i/2,2*i/2);
          g.setColor(c);
          g.fillRect( 30+10*i,35,20,80);

        try
          { Thread.sleep(500); } 
        catch(InterruptedException ex)
          { Thread.currentThread().interrupt(); }
        }
    }
}

class ProgressBar
{
    public static void main (String []args)
    {
        DrawPanel panel = new DrawPanel();
        JFrame app = new JFrame();
        app.setDefaultCloseOperation (JFrame.EXIT_ON_CLOSE);
        app.add(panel);
        app.setSize(500,200);
        app.setVisible(true);
    }
}  

非常感谢您的帮助,谢谢。

2 个答案:

答案 0 :(得分:6)

不要阻止EDT(事件调度线程) - 当发生这种情况时,GUI将“冻结”。而不是调用Thread.sleep(n)为重复的任务实现Swing Timer。有关详细信息,请参阅Concurrency in Swing。另外请务必查看由@Brian链接的进度条教程。它包含工作示例。

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

class DrawPanel extends JPanel
{
    int i = 0;
    public DrawPanel() {
        ActionListener animate = new ActionListener() {
            public void actionPerformed(ActionEvent ae) {
                repaint();
            }
        };
        Timer timer = new Timer(50,animate);
        timer.start();
    }
    public void paintComponent(Graphics g)
    {
        super.paintComponent(g);
        g.setColor(new Color(71,12,3));
        g.fillRect(35,30,410,90);

        Color c = new Color (12*i/2,8*i/2,2*i/2);
        g.setColor(c);
        g.fillRect( 30+10*i,35,20,80);

        i+=2;
        if (i>40) i = 0;
    }
}

class ProgressBar
{
    public static void main (String []args)
    {
        DrawPanel panel = new DrawPanel();
        JFrame app = new JFrame();
        app.setDefaultCloseOperation (JFrame.EXIT_ON_CLOSE);
        app.add(panel);
        app.setSize(500,200);
        app.setVisible(true);
    }
}

答案 1 :(得分:1)

我真的不会这样做。不应该像这样使用Swing刷新线程。你最好还是使用另一个线程(可能使用TimerTask)并根据需要重绘矩形。

查看Oracle ProgressBar tutorial以获取更多信息,代码等。