繁体   English   中英

为什么我绘制的图像在JPanel上闪烁?

[英]Why is my drawn image flickering on a JPanel?

完整的源代码在这个地址的 Github上。

我有一个JPanel,我在其中绘制一个背景和玩家精灵。 逻辑和重新绘制在不同的时间尺度上。 这是相关代码:

static long logicSleep = (long) (1e9 / 120);
static long drawSleep = (long) (1e9 / 60);
static long startTime;
static long logicTime, drawTime;

public static void main(String[] args) {
    startTime = System.nanoTime();
    while (keys[KeyEvent.VK_ESCAPE] == false) {
        if (System.nanoTime() > logicTime + startTime) {
            player.logic();
            logicTime += logicSleep;
        }
        if (System.nanoTime() > drawTime + startTime) {
            sheet.repaint();
            drawTime += drawSleep;
        }
    }
    System.exit(0);
}

因此,逻辑每1/120秒运行一次,绘图每1/60秒运行一次。 这按预期工作。 但是,图像经常闪烁。 这不是屏幕撕裂,因为我有Java 8并且我的显示器以60 Hz刷新(更不用说默认情况下如何在jpanel上启用双缓冲)。 闪烁似乎是位置的突然变化,尽管屏幕上角色的实际位置仅受到大约一个帧的影响。

这是我的自定义JPanel类:

static class Sheet extends JPanel {
    @Override
    public void paintComponent(Graphics g1) {
        super.paintComponent(g1);
        Graphics2D g = (Graphics2D) g1;
        g.translate((int) -(player.xPos + player.sprites[0].getWidth() / 2 - getWidth() / 2),
                (int) -(player.yPos + player.sprites[0].getHeight() / 2 - getHeight() / 2));
        g.drawImage(map, 0, 0, null);
        g.drawImage(player.sprites[player.spriteNum], (int) player.xPos, (int) player.yPos, null);
        g.drawImage(foreground, 0, 0, null);
    }
}

通过将逻辑睡眠改为每1/180秒运行一次,我设法解决了这个问题,但我很好奇为什么在特定时间尺度下运行逻辑会发生这种情况。 每1/240秒逻辑更新也会发生这种情况。

TL; DR:我的精灵因为我的逻辑滴答而闪烁,我不知道为什么。

你的主循环是一个忙等待循环。 它使CPU核心完全忙于进行比较( while条件和时序条件)。 这会浪费CPU,使您的系统响应性降低,通常不建议使用。 这本身可能会对其他线程的调度产生影响并导致闪烁。

此外,您选择了具有公分母(即60和180,60和120等)的更新时间。 这意味着存在应该触发两个条件的循环。 如果您的逻辑计算很长,则意味着重绘将延迟这么长时间。 这可能是闪烁的另一个原因。

执行定期任务的正确方法是使用Timer对象。 由于您有两个不同的时序循环,并且您需要它们是独立的,因此您应该使用两个单独的Timer实例。

对于Timer类,通常有两种选择:

  • javax.swing.Timer - 在Event Dispatch Thread中运行计划的操作。 这对于像repaint()这样的操作很有用,并且允许进行短逻辑操作,并且由于所有操作都在同一个线程中运行,因此不需要挥发性或其他方法来确保内存可见性。
  • java.util.Timer (或者, java.util.concurrent.ScheduledThreadPoolExecutor ) - 适合安排非GUI,更复杂的操作,你不想在EDT上运行。 如果使用此方法,则应注意使用SwingUtility方法更新GUI组件,并使用volatile或其他方法对EDT可见的数据进行更改,因为它们将在不同的线程中运行。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM