我在JPanel
的paint方法中遇到问题。
首先,我创建了一个扩展JFrame
的类。
然后我创建了一个扩展面板的类,并从该类创建一个对象并添加到框架中。
我想通过按键盘上的右键(VK_RIGHT
),在2秒内在面板上绘制小矩形。 (这意味着矩形清晰,并在一个新的位置绘制。我的问题是:我不能做定时绘画方法和重绘方法。我希望每两秒矩形前进5次(一步一步)但矩形继续只有一个
我的代码是:
Panel
import java.awt.Color;
import java.awt.Graphics;
import javax.swing.JPanel;
public class Panel extends JPanel{
private int x;
private int y;
public Panel()
{
x=100;
y=100;
this.setBackground(Color.RED);
}
//--------------------------------
@Override
public void paint(Graphics g)
{
super.paint(g);
g.setColor(Color.BLACK);
g.fillRect(x, y, 20, 20);
}
//--------------------------------
@Override
public void repaint()
{
super.repaint();
x+=20;
}
//--------------------------------
}
Frame
import java.awt.event.KeyEvent;
import java.awt.event.KeyListener;
import javax.swing.JFrame;
public class Frame extends JFrame {
private Panel p;
public Frame()
{
super("Test");
this.setBounds(1200, 300, 400, 400);
this.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
p=new Panel();
this.add(p);
this.addKeyListener(new KeyLis());
}
//----------------------------------
public static void main(String[] args) {
Frame a=new Frame();
a.setVisible(true);
}
//-----------------------------------
private class KeyLis implements KeyListener
{
@Override
public void keyPressed(KeyEvent arg0) {
if(arg0.getKeyCode() == KeyEvent.VK_RIGHT)
{
for(int i=0;i<5;i++)
{
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
p.repaint();
}
}
}
//---------------
@Override
public void keyReleased(KeyEvent arg0) {
// TODO Auto-generated method stub
}
//---------------
@Override
public void keyTyped(KeyEvent arg0) {
// TODO Auto-generated method stub
}
//---------------
}
//-----------------------------------------
}
答案 0 :(得分:2)
您的问题主要是for-loop
中的KeyListener
,它会阻止事件调度线程,阻止更新被绘制。
简单的解决方案是使用Swing Timer
,它在后台等待,触发EDT上的滴答,安全地更新ui并定期更新。
有关详细信息,请参阅Concurrency in Swing和How to use Swing Timers
如果由于某种原因,您发现Swing Timer
难以理解,您还可以尝试使用SwingWorker
,这将允许您使用后台线程循环,但是提供在EDT上下文中更新状态的方法。
有关详细信息,请参阅Worker Threads and SwingWorker
作为一般规则,覆盖paintComponent
而不是paint
。有关详细信息,请参阅Performing Custom Painting
不要使用repaint
更新您的状态,您可能不是唯一一个调用它的人,而是创建一个可以在视图和您的密钥控制器之间共享的模型。
请参阅Model-View-Controller和Observer Pattern了解一些想法
我也不鼓励使用KeyListener
并使用密钥绑定API
有关详细信息,请参阅How to Use Key Bindings
答案 1 :(得分:0)
如果您开始使用线程:
将您的KeyPressed事件更改为:
@Override
public void keyPressed(KeyEvent arg0) {
if (arg0.getKeyCode() == KeyEvent.VK_RIGHT) {
PaintThread th = new PaintThread(p);
th.start();
}
}
创建了一个主题:
import java.util.logging.Level;
import java.util.logging.Logger;
public class PaintThread extends Thread{
Panel panel;
PaintThread(Panel panel){
this.panel = panel;
}
@Override
public void run() {
for(int i = 0; i<5; i++){
try {
this.sleep(1000);
panel.repaint();
} catch (InterruptedException ex) {
Logger.getLogger(PaintThread.class.getName()).log(Level.SEVERE, null, ex);
}
}
}
}