有没有一种方法可以在不绘制多余形状的情况下调整JFrame的大小?

时间:2019-05-08 13:04:08

标签: java swing jframe awt drawing

我正在编写一个程序,该程序以一定间隔在窗口内绘制随机形状。我希望将形状存储在数组中,因为稍后需要从文本文件中检索有关每个形状的信息。只要我不尝试调整窗口大小,它就可以正常工作。每当我拉伸它时,就会出现很多新形状(比预期要快),可能是因为每次我调整大小时都会调用repaint(),但是我不知道如何防止它这样做。换句话说,如何防止剩余对象在调整大小时出现?

我将不胜感激任何技巧,不仅包括与我的问题直接相关的技巧,还包括我所做的任何编程选择。我是初学者。

我希望我说清楚了。如有需要,我们将乐意提供更多详细信息。

public Window() {
        setTitle("shapes");
        setSize(800, 600);
        setDefaultCloseOperation(WindowConstants.EXIT_ON_CLOSE);
        setBackground(Color.DARK_GRAY);
        setVisible(true);
    }

 public void paint(Graphics g) {
        addShape();
        g.clearRect(1,1, getWidth(), getHeight());
        for (Figure figure : myFigures) {
            figure.draw(g);
            System.out.println(figure.toString());
        }
    }

  public Thread drawThread() {
        return new Thread(() -> {
            while (true) {
                repaint();
                try {
                    Thread.sleep(1000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        });
    }

private void addShape() {
        int whichFigure = (int) (Math.random() * 3);
        switch (whichFigure) {
            case 1:
                myFigures.add(new Oval((int) (Math.random() * getWidth()), (int) (Math.random() * getHeight()), (int) (Math.random() * 100), (int) (Math.random() * 100)));
                System.out.println("oval added");
                break;
            case 2:
                myFigures.add(new Rectangle((int) (Math.random() * getWidth()), (int) (Math.random() * getHeight()), (int) (Math.random() * 100), (int) (Math.random() * 100)));
                System.out.println("rect added");
                break;
        }
    }

}

1 个答案:

答案 0 :(得分:3)

您不得通过绘画方法调用addShape。在发现时,您无法控制何时调用绘画方法。调整大小不是唯一可以触发绘画的内容。移动或抬起窗户可以。即使将鼠标移到窗口上也可以。切勿使用绘画方法更改数据。

在另一个位置更改数据,例如在调用repaint()之前。所有Swing操作都需要在AWT事件分发线程中进行,并且Swing操作使用的数据也需要在该线程中进行修改(除非您采取措施使代码成为线程安全的,例如同步)。在该线程中定期执行代码的最简单方法是使用javax.swing.Timer类:

public Timer drawThread() {
    return new Timer(1000, new ActionListener() {
        @Override
        public void actionPerformed(ActionEvent event) {
            addShape();
            repaint();
        }
    });
}

请注意,类似于线程,计时器需要调用其start()(或restart())方法。