gpt4 book ai didi

java - repaint() 在 keylistener 之外不起作用

转载 作者:行者123 更新时间:2023-12-02 11:41:35 27 4
gpt4 key购买 nike

在过去的一周左右的时间里,我一直在开发一款游戏,在游戏中,你是一个白色方 block ,试图躲避从屏幕顶部掉落的红色方 block 。你只能向左和向右移动,只有 3 个可用的位置可以移动到(左、中、右),并且随着游戏的进行,红色方 block 开始下落得更快。当我移动玩家(白色方 block )时,一切都会重新绘制,但是当红色方 block 的位置更新时,它根本不会重新绘制。假设红色方 block 每秒移动一次;如果我等一秒钟,重画不会做任何事情。但是,如果我等待一秒钟(或任意时间)然后移动玩家,两个方 block 都会立即跳到新位置。浏览了一个小时的代码后,我仍然找不到任何问题,而且我对编码还很陌生,这无济于事,这可能意味着这是一个非常明显的错误。

import java.util.*;
import javax.swing.*;
import java.awt.*;
import java.awt.event.KeyEvent;
import java.awt.event.KeyListener;
import java.awt.Color;
import java.awt.Graphics;
import java.awt.Graphics2D;
import javax.swing.JComponent;
import javax.swing.JFrame;

public class boxface extends JComponent implements KeyListener {

private static int x=0, y=0;
static int counter = 0;
static Thread t = new Thread();

public void keyPressed(KeyEvent e) {
if(e.getKeyCode()== KeyEvent.VK_RIGHT)
moveRight();
else if(e.getKeyCode()== KeyEvent.VK_LEFT)
moveLeft(); }
public void keyReleased(KeyEvent e) {}
public void keyTyped(KeyEvent e) {}

static Rectangle player = new Rectangle(x, 400, 50, 50);
Rectangle bg = new Rectangle(0, 0, 700, 750);
static int x2 = ((int)((Math.random()*3)))*50;
static Rectangle enemy = new Rectangle(x2, y, 50, 50);

public void paintComponent(Graphics g) {
super.paintComponent(g);
Graphics2D g2 = (Graphics2D) g;
g2.setColor(Color.BLACK);
g2.fill(bg);
g2.setColor(Color.WHITE);
g2.fill(player);
g2.setColor(Color.RED);
g2.fill(enemy); }

public void moveLeft() {
if(x > 0) {
x -= 50;
player.setLocation(x, 400);
repaint();}}

public void moveRight() {
if(x < 100) {
x += 50;
player.setLocation(x, 400);
repaint();} }

public void enemyDown() {
if(enemy.getBounds().y == 400) {
x2=((int)((Math.random()*3)))*50;
y=0;
enemy.setLocation(x2, y);
counter++;
repaint(); }
else {
y = y + 50;
enemy.setLocation(x2, y);
repaint();}}

public boxface(){
addKeyListener(this);
setFocusable(true);
setFocusTraversalKeysEnabled(false); }

public static void main(String[] args) {
javax.swing.SwingUtilities.invokeLater(new Runnable() {
public void run() {
JFrame f = new JFrame();
f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
f.setBounds(400, 200, 156, 479);
f.setMinimumSize(new Dimension(156, 0));
f.setResizable(false);
f.getContentPane().add(new boxface());
f.setVisible(true); }
});

boxface exec = new boxface();
int t = 20;

for(long i = 0; i < 21; i++) {
if(i == t) {
exec.enemyDown();
i = 0; }
if(counter == 10) t = 15;
if(counter == 25) t = 10;
if(counter == 50) t = 7;
if(counter == 100) t = 5;
if(counter == 150) t = 3;
if(counter == 225) t = 1;
if(enemy.intersects(player))
break;
System.out.println(i); }
}
}

最佳答案

Swing 使用“被动渲染”算法。这意味着 UI 将根据 RepaintManager

做出的决定不定期更新

See Painting in AWT and Swing了解更多详情

您需要的是某种方式来安排定期更新。在讨论这个之前,先介绍一些基本的博弈论......

游戏/主循环...

大多数游戏都有“主循环”的概念。 “主循环”执行许多工作,它:

  • 更新游戏状态。这包括:
    • 根据输入(键盘/鼠标/操纵杆)的当前状态更新玩家的位置
    • 更新世界中其他物体/障碍物的位置
    • 碰撞检测
    • 其他需要更新才能渲染的状态
  • 渲染当前状态

虽然您可以使用 Thread 来完成此操作,但 Swing 不是线程安全的,这意味着您永远不应该在上下文之外更新 UI 的状态(或 UI 依赖的任何状态)事件调度线程的。 Swing 也是单线程的,因此您也不能简单地使用 Thread.sleep 或事件调度线程上下文中的其他循环。

最简单的解决方案是使用 Swing Timer。这是伪循环,在指定的延迟后重复调用。每次触发计时器时,您都会执行“主循环”操作并调用repaint,从而安排对 UI 进行(大部分)定期更新。

参见How to use Swing Timers了解更多详情

其他考虑因素...

由于游戏状态实际上是在“主循环”内更新的,因此您无法再直接响应关键事件。相反,您需要设置一系列标志来指示输入触发器的当前状态(即最简单形式的 isAPressed)。

在大多数情况下,这将调用Set和预定义的一系列输入。我喜欢使用enum,因为它清楚地定义了接受的输入(上/下/左/右等...)。按下时,您将适当的输入添加到 Set 中,释放时,将其删除。它处理任何重复的状态,并消除第一次按下和重复的按键事件之间的延迟的奇怪现象。

谈论重大事件。 KeyListener 存在已知问题,虽然有“黑客”可以“解决”它,但它们只是“黑客”。

监视(有限)按键输入的推荐方法是使用 key bindings API

关于java - repaint() 在 keylistener 之外不起作用,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/48513803/

27 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com