Java 2D игровая графика - PullRequest
       15

Java 2D игровая графика

15 голосов
/ 26 декабря 2009

В следующем семестре у нас есть модуль для создания Java-приложений в команде. Требование модуля - сделать игру. В рождественские каникулы я немного потренировался, но не могу придумать лучший способ рисовать графику.

Я использую объект Java Graphics2D для рисования фигур на экране и вызываю repaint() 30 раз в секунду, но это ужасно мерцает. Есть ли лучший способ рисовать высокопроизводительную 2D-графику на Java?

Ответы [ 5 ]

17 голосов
/ 26 декабря 2009

То, что вы хотите сделать, это создать компонент canvas с BufferStrategy и выполнить рендеринг, код ниже должен показать вам, как это работает. Я извлек код из моего самописного движка над здесь .

Производительность зависит исключительно от того, что вы хотите нарисовать, мои игры в основном используют изображения. Приблизительно с 1500 из них я все еще выше 200 FPS в 480x480. И только с 100 изображениями я удаляю 6k FPS при отключении ограничения кадра.

Небольшую игру (в этой игре около 120 изображений одновременно на экране), которую я создал, можно найти здесь (да, приведенный ниже подход также отлично работает в качестве апплета.)

import java.awt.Canvas;
import java.awt.Color;
import java.awt.Graphics2D;
import java.awt.GraphicsConfiguration;
import java.awt.GraphicsEnvironment;
import java.awt.Toolkit;
import java.awt.Transparency;
import java.awt.event.WindowAdapter;
import java.awt.event.WindowEvent;
import java.awt.image.BufferStrategy;
import java.awt.image.BufferedImage;

import javax.swing.JFrame;
import javax.swing.WindowConstants;

public class Test extends Thread {
    private boolean isRunning = true;
    private Canvas canvas;
    private BufferStrategy strategy;
    private BufferedImage background;
    private Graphics2D backgroundGraphics;
    private Graphics2D graphics;
    private JFrame frame;
    private int width = 320;
    private int height = 240;
    private int scale = 1;
    private GraphicsConfiguration config =
            GraphicsEnvironment.getLocalGraphicsEnvironment()
                .getDefaultScreenDevice()
                .getDefaultConfiguration();

    // create a hardware accelerated image
    public final BufferedImage create(final int width, final int height,
            final boolean alpha) {
        return config.createCompatibleImage(width, height, alpha
                ? Transparency.TRANSLUCENT : Transparency.OPAQUE);
    }

    // Setup
    public Test() {
        // JFrame
        frame = new JFrame();
        frame.addWindowListener(new FrameClose());
        frame.setDefaultCloseOperation(WindowConstants.DO_NOTHING_ON_CLOSE);
        frame.setSize(width * scale, height * scale);
        frame.setVisible(true);

        // Canvas
        canvas = new Canvas(config);
        canvas.setSize(width * scale, height * scale);
        frame.add(canvas, 0);

        // Background & Buffer
        background = create(width, height, false);
        canvas.createBufferStrategy(2);
        do {
            strategy = canvas.getBufferStrategy();
        } while (strategy == null);
        start();
    }

    private class FrameClose extends WindowAdapter {
        @Override
        public void windowClosing(final WindowEvent e) {
            isRunning = false;
        }
    }

    // Screen and buffer stuff
    private Graphics2D getBuffer() {
        if (graphics == null) {
            try {
                graphics = (Graphics2D) strategy.getDrawGraphics();
            } catch (IllegalStateException e) {
                return null;
            }
        }
        return graphics;
    }

    private boolean updateScreen() {
        graphics.dispose();
        graphics = null;
        try {
            strategy.show();
            Toolkit.getDefaultToolkit().sync();
            return (!strategy.contentsLost());

        } catch (NullPointerException e) {
            return true;

        } catch (IllegalStateException e) {
            return true;
        }
    }

    public void run() {
        backgroundGraphics = (Graphics2D) background.getGraphics();
        long fpsWait = (long) (1.0 / 30 * 1000);
        main: while (isRunning) {
            long renderStart = System.nanoTime();
            updateGame();

            // Update Graphics
            do {
                Graphics2D bg = getBuffer();
                if (!isRunning) {
                    break main;
                }
                renderGame(backgroundGraphics); // this calls your draw method
                // thingy
                if (scale != 1) {
                    bg.drawImage(background, 0, 0, width * scale, height
                            * scale, 0, 0, width, height, null);
                } else {
                    bg.drawImage(background, 0, 0, null);
                }
                bg.dispose();
            } while (!updateScreen());

            // Better do some FPS limiting here
            long renderTime = (System.nanoTime() - renderStart) / 1000000;
            try {
                Thread.sleep(Math.max(0, fpsWait - renderTime));
            } catch (InterruptedException e) {
                Thread.interrupted();
                break;
            }
            renderTime = (System.nanoTime() - renderStart) / 1000000;

        }
        frame.dispose();
    }

    public void updateGame() {
        // update game logic here
    }

    public void renderGame(Graphics2D g) {
        g.setColor(Color.BLACK);
        g.fillRect(0, 0, width, height);
    }

    public static void main(final String args[]) {
        new Test();
    }
}
8 голосов
/ 26 декабря 2009

Мерцание происходит из-за того, что вы пишете прямо на экран. Используйте буфер для рисования, а затем запишите весь экран за один раз. Это Double Buffering, о котором вы, возможно, слышали раньше. Здесь - самая простая из возможных форм.

public void paint(Graphics g)
{

    Image image = createImage(size + 1, size + 1);
    Graphics offG = image.getGraphics();
    offG.setColor(Color.BLACK);
    offG.fillRect(0, 0, getWidth(), getHeight());
    // etc

См. Использование закадровой графики offG. Создание изображения вне экрана стоит дорого, поэтому я бы предложил создать его только при первом вызове.

Есть и другие области, в которых вы можете улучшить это, например, , создав совместимое изображение , используя отсечение и т. Д. Для более тонкой настройки управления анимацией вы должны изучить активный рендеринг .

Есть приличная страница, которую я добавил в закладки для обсуждения учебных пособий здесь .

Удачи!

3 голосов
/ 26 декабря 2009

Java OpenGL ( JOGL ) - это один из способов.

2 голосов
/ 26 декабря 2009

Я думаю, вы сделали переопределение из paint(Graphics g)? Это не хороший способ. Используйте тот же код, но в paintComponent(Graphics g) вместо paint(Graphics g).

Метка, которую вы можете искать - doublebuffer. Это то, что будет сделано автоматически путем переопределения paintComponent.

0 голосов
/ 12 сентября 2012

Существует простой способ оптимизировать вашу программу. Избавьтесь от любого сложного кода и просто используйте JComponent вместо Canvas и нарисуйте на нем свои объекты. Это все. Наслаждайся этим ...

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...