Проблемы с javax.swing.timer при реализации реплики Flappy Bird в Java

Я пишу точную копию игры Flappy Bird на Java, но я сталкиваюсь с некоторыми вещами, касающимися потоков. По сути, у меня есть 4 класса в моем проекте. Renderer класс, который помогает мне визуализировать игру и выглядеть так:

import java.awt.Graphics;

import javax.swing.JPanel;

public class Renderer extends JPanel
{

    private static final long serialVersionUID = 1L;

    @Override
    protected void paintComponent(Graphics g)
    {
        super.paintComponent(g);

        FlappyBird.flappyBird.repaint(g);
    }

}

у меня есть Database класс, который помогает мне подключиться к локальной размещенной базе данных, чтобы сохранить результаты, и это выглядит так:

import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.PreparedStatement;
import java.sql.ResultSet;  
import java.sql.SQLException;
import java.sql.Statement;

public class Database  {

    private static Connection con;
    private String driverName = "com.mysql.jdbc.Driver";

    public Database(String host, String username, String password){
       try {

            con = DriverManager.getConnection( host, username, password );

           }

       catch (SQLException e) {
        System.out.println(e.getMessage());
        e.printStackTrace();
       }

       try {
            Class.forName(driverName).newInstance();
        } catch (InstantiationException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        } catch (ClassNotFoundException e) {
            System.err.println(e.getMessage() + "------Cannot Load Driver");

        }
   }

    public void insertRow (Row row)
   {
        try {         

            PreparedStatement stmt = con.prepareStatement("insert into Scores (Username, Score, CoinScore) VALUES (?, ?, ?);");
            stmt.setString(1, row.getUserName());
            stmt.setInt(2, row.getScore());
            stmt.setInt(3, row.getCoinScore());


            stmt.executeUpdate();
        } catch (SQLException e) {
            System.out.println(e.getMessage());
            e.printStackTrace();
        }
    }
}

у меня есть Row класс, представляющий одну строку, которая собирается быть вставленной в базу данных, и ее файл выглядит следующим образом:

public class Row {

    private String username;
    private int score;
    private int coinscore;

    public Row(String username, int score, int coinscore)
    {
        this.username = username;
        this.score = score;
        this.coinscore = coinscore;
    }

    public String getUserName()
    {
        return this.username;
    }

    public int getScore()
    {
        return this.score;
    }

    public int getCoinScore()
    {
        return this.coinscore;
    }
} 

И у меня есть основной класс по имени Flappy Bird которая является самой игрой и выглядит так:

import java.awt.Color;
import java.awt.Font;
import java.awt.Graphics;
import java.awt.Image;
import java.awt.Rectangle;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.awt.event.KeyEvent;
import java.awt.event.KeyListener;
import java.awt.event.MouseEvent;
import java.awt.event.MouseListener;
import java.awt.image.*;
import java.util.ArrayList;
import java.util.Random;
import java.util.Scanner;

import javax.swing.JFrame;
import javax.swing.Timer;


public class FlappyBird implements ActionListener, MouseListener, KeyListener
{

    public static FlappyBird flappyBird;

    public final int WIDTH = 800, HEIGHT = 800;

    public Renderer renderer;

    public Rectangle bird;

    public ArrayList<Rectangle> columns,coins;

    public int ticks, yMotion, score, coinScore;

    public boolean gameOver, started;

    public Random rand;

    public Timer timer;

    public FlappyBird()
    {
        JFrame jframe = new JFrame();
        timer = new Timer(50, this);

        renderer = new Renderer();
        rand = new Random();

        jframe.add(renderer);
        jframe.setTitle("Flappy Bird");
        jframe.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        jframe.setSize(WIDTH, HEIGHT);
        jframe.addMouseListener(this);
        jframe.addKeyListener(this);
        jframe.setResizable(false);
        jframe.setVisible(true);

        bird = new Rectangle(WIDTH / 2 - 10, HEIGHT / 2 - 10, 20, 20);
        columns = new ArrayList<Rectangle>();
        coins = new ArrayList<Rectangle>();

        addColumn(true);
        addColumn(true);
        addColumn(true);
        addColumn(true);

        timer.start();
    }

    public void addColumn(boolean start)
    {
        int space = 300;
        int width = 100;
        int height = 50 + rand.nextInt(300);

        if (start)
        {
            columns.add(new Rectangle(WIDTH + width + columns.size() * 300, HEIGHT - height - 120, width, height));
            coins.add(new Rectangle(WIDTH+10+columns.size()*300,HEIGHT-height-100,15,15));
            columns.add(new Rectangle(WIDTH + width + (columns.size() - 1) * 300, 0, width, HEIGHT - height - space));
        }
        else
        {
            columns.add(new Rectangle(columns.get(columns.size() - 1).x + 600, HEIGHT - height - 120, width, height));
            coins.add(new Rectangle(columns.get(columns.size()-1).x+rand.nextInt(600) + 100,HEIGHT -height - rand.nextInt(100),15,15));
            columns.add(new Rectangle(columns.get(columns.size() - 1).x, 0, width, HEIGHT - height - space));
        }
    }

    public void paintColumn(Graphics g, Rectangle column)
    {
        g.setColor(Color.green.darker());
        g.fillRect(column.x, column.y, column.width, column.height);
    }

    public void paintCoin(Graphics g, Rectangle coin)
    {
        g.setColor(Color.yellow.darker());
        g.fillRect(coin.x, coin.y, coin.width, coin.height);
    }

    public void paintCoinCyan(Graphics g, Rectangle coin)
    {
        g.setColor(Color.cyan);
        g.fillRect(coin.x, coin.y, coin.width, coin.height);
    }

    public void jump()
    {
        if (gameOver)
        {
            bird = new Rectangle(WIDTH / 2 - 10, HEIGHT / 2 - 10, 20, 20);
            columns.clear();
            coins.clear();
            yMotion = 0;
            score = 0;
            coinScore = 0;

            addColumn(true);
            addColumn(true);
            addColumn(true);
            addColumn(true);
           Scanner in = new Scanner(System.in);
           Database myDB = new Database("jdbc:mysql://localhost:3306/Leaderboard","root","greenlantern10");
           System.out.println("Input your Username: ");
           String user = in.next();
           in.close();

           Row row = new Row(user,score,coinScore);
           myDB.insertRow(row);
            gameOver = false;


        }

        if (!started)
        {
            started = true;
        }
        else if (!gameOver)
        {
            if (yMotion > 0)
            {
                yMotion = 0;
            }

            yMotion -= 10;
        }
    }



    @Override
    public void actionPerformed(ActionEvent e)
    {
        int speed = 15;

        ticks++;

        if (started)
        {
            for (int i = 0; i < columns.size(); i++)
            {
                Rectangle column = columns.get(i);

                column.x -= speed;

            }

            for (int i = 0; i< coins.size();i++)
            {
                Rectangle coin = coins.get(i);

                coin.x -= speed;

            }

            if (ticks % 2 == 0 && yMotion < speed)
            {
                yMotion += 2;
            }

            for (int i = 0; i < columns.size(); i++)
            {
                Rectangle column = columns.get(i);

                if (column.x + column.width < 0)
                {
                    columns.remove(column);

                    if (column.y == 0)
                    {
                        addColumn(false);
                    }
                }
            }

            bird.y += yMotion;

            for (Rectangle column : columns)
            {
                if (column.y == 0 && bird.x + bird.width / 2 > column.x + column.width / 2 - 10 && bird.x + bird.width / 2 < column.x + column.width / 2 + 10)
                {
                    score++;
                }

                if (column.intersects(bird))
                {
                    gameOver = true;

                    if (bird.x <= column.x)
                    {
                        bird.x = column.x - bird.width;

                    }
                    else
                    {
                        if (column.y != 0)
                        {
                            bird.y = column.y - bird.height;
                        }
                        else if (bird.y < column.height)
                        {
                            bird.y = column.height;
                        }
                    }
                }
            }

            for (Rectangle coin : coins)
            {
                if (bird.intersects(coin))
                {
                  coinScore++;

                }
            }

            if (bird.y > HEIGHT - 120)
            {
                bird.y = HEIGHT - 120;
            }

            if (bird.y < 0)
            {
                gameOver = true;
            }

            if (bird.y + yMotion >= HEIGHT - 120)
            {
                bird.y = HEIGHT - 120 - bird.height;
                gameOver = true;
            }
        }

        renderer.repaint();
    }

    public void repaint(Graphics g)
    {
        g.setColor(Color.cyan);
        g.fillRect(0, 0, WIDTH, HEIGHT);

        g.setColor(Color.orange);
        g.fillRect(0, HEIGHT - 120, WIDTH, 120);

        g.setColor(Color.green);
        g.fillRect(0, HEIGHT - 120, WIDTH, 20);

        g.setColor(Color.red);
        g.fillRect(bird.x, bird.y, bird.width, bird.height);

        for (Rectangle column : columns)
        {
            paintColumn(g, column);
        }

        for (Rectangle coin : coins)
        {
            paintCoin(g,coin);
        }

        g.setColor(Color.white);
        g.setFont(new Font("SANS_SERIF", 1, 100));

        if (!started)
        {
            g.drawString("Click or press SPACE to start!", 40, HEIGHT / 2 - 50);
        }

        if (gameOver)
        {
            g.drawString("Game Over!", 200, HEIGHT / 2 - 60);
        }

        if (!gameOver && started)
        {
            g.drawString("Score: "+String.valueOf(score), WIDTH / 2 - 400 , 150);
            g.drawString("Coin score: "+String.valueOf(coinScore),WIDTH / 2- 400, 50);
        }
    }

    public static void main(String[] args)
    {
        flappyBird = new FlappyBird();
    }

    @Override
    public void mouseClicked(MouseEvent e)
    {
        jump();
    }

    @Override
    public void keyReleased(KeyEvent e)
    {
        if (e.getKeyCode() == KeyEvent.VK_SPACE)
        {
            jump();
        }
    }

    @Override
    public void mousePressed(MouseEvent e)
    {
    }

    @Override
    public void mouseReleased(MouseEvent e)
    {
    }

    @Override
    public void mouseEntered(MouseEvent e)
    {
    }

    @Override
    public void mouseExited(MouseEvent e)
    {
    }

    @Override
    public void keyTyped(KeyEvent e)
    {

    }

    @Override
    public void keyPressed(KeyEvent e)
    {

    }

}

Проблема возникает, когда я пытаюсь вставить новую строку в базу данных (когда установлена ​​переменная gameover, т.е. gameover = true) . Судя по всему, приложение правильно выполняет вставку в базу данных, но когда я пытаюсь воспроизвести игру, я получаю некоторые исключения, касающиеся Event Dispatch Thread, и игра больше не продолжается. До сих пор я пытался остановить таймер, пока я вставляю данные в базу данных, а затем запускаю их заново, но это не сработало. Каково ваше мнение, что я должен сделать, чтобы решить эту проблему?

PS: кроме этой проблемы, игра работает без каких-либо недостатков.

Трассировка стека:

Exception in thread "AWT-EventQueue-0" java.util.NoSuchElementException
    at java.util.Scanner.throwFor(Unknown Source)
    at java.util.Scanner.next(Unknown Source)
    at flappyBird.FlappyBird.jump(FlappyBird.java:128)
    at flappyBird.FlappyBird.keyReleased(FlappyBird.java:322)
    at java.awt.Component.processKeyEvent(Unknown Source)
    at java.awt.Component.processEvent(Unknown Source)
    at java.awt.Container.processEvent(Unknown Source)
    at java.awt.Window.processEvent(Unknown Source)
    at java.awt.Component.dispatchEventImpl(Unknown Source)
    at java.awt.Container.dispatchEventImpl(Unknown Source)
    at java.awt.Window.dispatchEventImpl(Unknown Source)
    at java.awt.Component.dispatchEvent(Unknown Source)
    at java.awt.KeyboardFocusManager.redispatchEvent(Unknown Source)
    at java.awt.DefaultKeyboardFocusManager.dispatchKeyEvent(Unknown Source)
    at java.awt.DefaultKeyboardFocusManager.preDispatchKeyEvent(Unknown Source)
    at java.awt.DefaultKeyboardFocusManager.typeAheadAssertions(Unknown Source)
    at java.awt.DefaultKeyboardFocusManager.dispatchEvent(Unknown Source)
    at java.awt.Component.dispatchEventImpl(Unknown Source)
    at java.awt.Container.dispatchEventImpl(Unknown Source)
    at java.awt.Window.dispatchEventImpl(Unknown Source)
    at java.awt.Component.dispatchEvent(Unknown Source)
    at java.awt.EventQueue.dispatchEventImpl(Unknown Source)
    at java.awt.EventQueue.access$500(Unknown Source)
    at java.awt.EventQueue$3.run(Unknown Source)
    at java.awt.EventQueue$3.run(Unknown Source)
    at java.security.AccessController.doPrivileged(Native Method)
    at java.security.ProtectionDomain$JavaSecurityAccessImpl.doIntersectionPrivilege(Unknown Source)
    at java.security.ProtectionDomain$JavaSecurityAccessImpl.doIntersectionPrivilege(Unknown Source)
    at java.awt.EventQueue$4.run(Unknown Source)
    at java.awt.EventQueue$4.run(Unknown Source)
    at java.security.AccessController.doPrivileged(Native Method)
    at java.security.ProtectionDomain$JavaSecurityAccessImpl.doIntersectionPrivilege(Unknown Source)
    at java.awt.EventQueue.dispatchEvent(Unknown Source)
    at java.awt.EventDispatchThread.pumpOneEventForFilters(Unknown Source)
    at java.awt.EventDispatchThread.pumpEventsForFilter(Unknown Source)
    at java.awt.EventDispatchThread.pumpEventsForHierarchy(Unknown Source)
    at java.awt.EventDispatchThread.pumpEvents(Unknown Source)
    at java.awt.EventDispatchThread.pumpEvents(Unknown Source)
    at java.awt.EventDispatchThread.run(Unknown Source)

1 ответ

Решение

В подпрограмме Jump вы потеряли код. Он в основном пытается найти вход игрока (имя) в прыжке.

По вашей логике, вы дважды выполняете эту процедуру, а во второй раз она терпит неудачу. Скорее всего, это сбой из-за ввода System.in, который собирался между играми.

Либо промойте свой System.in, прежде чем запрашивать его, или, что еще лучше, используйте библиотеки Swing, чтобы фактически запросить имя игрока в игре.

Другие вопросы по тегам