手把手一步一步教你使用Java开发一个大型街机动作闯关类游戏01游戏窗口

简介: 手把手一步一步教你使用Java开发一个大型街机动作闯关类游戏01游戏窗口

项目源码

项目源码

游戏配置信息类

Config.java 没什么解释的。

package config;

public class Config {
    public final static String TITEL = "fight_to_the_end";
    public final static String VERSION = "v1.0";
    
    public final static byte BUFFERS = 2;
    public final static int FPS = 60;

    public final static boolean DEBUG = true;
}

主函数Main.java

package main;


public class Main
{ 
    public static void main(String  [] args)
    {
        GameApp app = new GameApp();
    }

}

创建了一个GameApp对象,GameApp对象是我们游戏的主循环。
GameApp.java

package main;

import config.Config;

import javax.swing.*;
import java.awt.*;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.awt.image.BufferStrategy;


public class GameApp {
    private boolean _gameRunning = true;
    private Frame _frm;

    public GameApp(){
        try{
            _frm = new Frame();
            _frm.setUndecorated(true);
            _frm.setIgnoreRepaint(true);
            _frm.setTitle("");

            JButton button = new JButton("close");
            button.addActionListener(new ActionListener() {
                public void actionPerformed(ActionEvent e) {
                    _gameRunning = false;
                    _frm.dispose();
                }
            });
            _frm.add(button);
            _frm.setSize(800,600);
            _frm.setLocation(100, 100);
            _frm.setVisible(true);
            _frm.setResizable(false);
            _frm.createBufferStrategy(Config.BUFFERS);
            _gameLoop();

        }catch (Exception e) {
            e.printStackTrace();
        }finally{
            System.exit(0);
        }
    }

GameApp构造函数主要做了2件事:
1.创建游戏窗口;
2.启动game主循环_gameLoop()

            _frm = new Frame();
            _frm.setUndecorated(true);
            _frm.setIgnoreRepaint(true);
            _frm.setTitle("");

            JButton button = new JButton("close");
            button.addActionListener(new ActionListener() {
                public void actionPerformed(ActionEvent e) {
                    _gameRunning = false;
                    _frm.dispose();
                }
            });
            _frm.add(button);
            _frm.setSize(800,600);
            _frm.setLocation(100, 100);
            _frm.setVisible(true);
            _frm.setResizable(false);
            _frm.createBufferStrategy(Config.BUFFERS);
            _gameLoop();

创建窗体,设置窗体大小,位置,可见性,窗口上添加了一个大按钮,点击后可以关闭本窗体。

其中比较重要的一行代码是:

 _frm.createBufferStrategy(Config.BUFFERS);

设置窗体的缓冲策略为双缓冲。就是我们先将游戏每一帧(包含很多图片,我们下一节详细介绍)的全部数据,先全部画在内存中,再一次性的粘贴到屏幕上。这样可以提高绘制效率,减少屏幕闪烁。
接下来我们看_gameLoop()方法:

    private void _gameLoop(){
        BufferStrategy buff = _frm.getBufferStrategy();
        while(_gameRunning){
            Graphics2D g = (Graphics2D)buff.getDrawGraphics();

            // Rendering
            _initRendering(g);

            if(Config.DEBUG){
                _displayInfoText(g);
            }

            g.dispose();

            if (!buff.contentsLost())
            {
                buff.show();
            }
            Toolkit.getDefaultToolkit().sync();

            try {
                Thread.sleep(1);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
    private void _initRendering(Graphics2D g){
        g.setColor(Color.black);
        g.fillRect(0, 0, 800, 600);
    }

    private void _displayInfoText(Graphics2D g){
        g.setColor(Color.white);
        g.drawString(Config.TITEL+ " "+ Config.VERSION, 20, 20);
    }

该方法整体结构是一个死循环,每次循环Thread.sleep(1); 让主线程睡眠1毫秒,让出cpu时间片;让其他进程得到执行,防止cpu使用率过高。

Graphics2D g = (Graphics2D)buff.getDrawGraphics();

            // Rendering
            _initRendering(g);

            if(Config.DEBUG){
                _displayInfoText(g);
            }

            g.dispose();

在内存中描画:填充一个黑色矩形,并且显示一个白色的字符串。

if (!buff.contentsLost())
            {
                buff.show();
            }
            Toolkit.getDefaultToolkit().sync();

将内存中的图像,粘贴到屏幕上。

本节最终效果

运行程序,显示一个黑窗体,点击任意位置,窗体关闭。

468490-be9a46a8b3583aaa.png

目录
相关文章
|
26天前
|
移动开发 前端开发 Java
Java最新图形化界面开发技术——JavaFx教程(含UI控件用法介绍、属性绑定、事件监听、FXML)
JavaFX是Java的下一代图形用户界面工具包。JavaFX是一组图形和媒体API,我们可以用它们来创建和部署富客户端应用程序。 JavaFX允许开发人员快速构建丰富的跨平台应用程序,允许开发人员在单个编程接口中组合图形,动画和UI控件。本文详细介绍了JavaFx的常见用法,相信读完本教程你一定有所收获!
Java最新图形化界面开发技术——JavaFx教程(含UI控件用法介绍、属性绑定、事件监听、FXML)
|
12天前
|
监控 JavaScript 数据可视化
建筑施工一体化信息管理平台源码,支持微服务架构,采用Java、Spring Cloud、Vue等技术开发。
智慧工地云平台是专为建筑施工领域打造的一体化信息管理平台,利用大数据、云计算、物联网等技术,实现施工区域各系统数据汇总与可视化管理。平台涵盖人员、设备、物料、环境等关键因素的实时监控与数据分析,提供远程指挥、决策支持等功能,提升工作效率,促进产业信息化发展。系统由PC端、APP移动端及项目、监管、数据屏三大平台组成,支持微服务架构,采用Java、Spring Cloud、Vue等技术开发。
|
1月前
|
存储 JavaScript 前端开发
基于 SpringBoot 和 Vue 开发校园点餐订餐外卖跑腿Java源码
一个非常实用的校园外卖系统,基于 SpringBoot 和 Vue 的开发。这一系统源于黑马的外卖案例项目 经过站长的进一步改进和优化,提供了更丰富的功能和更高的可用性。 这个项目的架构设计非常有趣。虽然它采用了SpringBoot和Vue的组合,但并不是一个完全分离的项目。 前端视图通过JS的方式引入了Vue和Element UI,既能利用Vue的快速开发优势,
128 13
|
1月前
|
前端开发 Java 测试技术
java日常开发中如何写出优雅的好维护的代码
代码可读性太差,实际是给团队后续开发中埋坑,优化在平时,没有那个团队会说我专门给你一个月来优化之前的代码,所以在日常开发中就要多注意可读性问题,不要写出几天之后自己都看不懂的代码。
66 2
|
IDE 小程序 前端开发
详细解读java的俄罗斯方块游戏的源代码--【课程设计】
详细解读java的俄罗斯方块游戏的源代码--【课程设计】
|
Java 定位技术 开发者
基于Java的俄罗斯方块游戏
基于Java的俄罗斯方块游戏
|
Java 定位技术
Java---俄罗斯方块小游戏
去年就已经学了这个技术了,一直没去写,现在抽个时间写了个俄罗斯方块游戏。 只有简单的新游戏,暂停,继续,积分功能。简单的实现了俄罗斯的经典功能。 不介绍了,有兴趣的自己运行一下,后面贴出了图片。
1026 0
|
15天前
|
监控 Java
java异步判断线程池所有任务是否执行完
通过上述步骤,您可以在Java中实现异步判断线程池所有任务是否执行完毕。这种方法使用了 `CompletionService`来监控任务的完成情况,并通过一个独立线程异步检查所有任务的执行状态。这种设计不仅简洁高效,还能确保在大量任务处理时程序的稳定性和可维护性。希望本文能为您的开发工作提供实用的指导和帮助。
71 17
|
25天前
|
Java
Java—多线程实现生产消费者
本文介绍了多线程实现生产消费者模式的三个版本。Version1包含四个类:`Producer`(生产者)、`Consumer`(消费者)、`Resource`(公共资源)和`TestMain`(测试类)。通过`synchronized`和`wait/notify`机制控制线程同步,但存在多个生产者或消费者时可能出现多次生产和消费的问题。 Version2将`if`改为`while`,解决了多次生产和消费的问题,但仍可能因`notify()`随机唤醒线程而导致死锁。因此,引入了`notifyAll()`来唤醒所有等待线程,但这会带来性能问题。
Java—多线程实现生产消费者