如何用Java还原童年回忆?在线教你完成贪吃蛇小游戏
myzbx 2025-05-16 15:02 2 浏览
今天我就从零开始来完成这个小游戏,完成的方式也是一步一步的添加功能这样的方式来实现。
额,不好意思,放错了,重来
第一步完成的功能:写一个界面
大家见到的贪吃蛇小游戏,界面肯定是少不了的。因此,第一步就是写一个小界面。
实现代码如下:
public class SnakeFrame extends Frame{
//方格的宽度和长度
public static final int BLOCK_WIDTH = 15 ;
public static final int BLOCK_HEIGHT = 15 ;
//界面的方格的行数和列数
public static final int ROW = 40;
public static final int COL = 40;
public static void main(String[] args) {
new SnakeFrame().launch();
}
public void launch(){
this.setTitle("Snake");
this.setSize(ROW*BLOCK_HEIGHT, COL*BLOCK_WIDTH);
this.setLocation(300, 400);
this.addWindowListener(new WindowAdapter() {
@Override
public void windowClosing(WindowEvent e) {
System.exit(0);
}
});
this.setResizable(false);
this.setVisible(true);
}
}
第二步完成的功能:在界面上画成一格一格的
我们见过的贪吃蛇游戏,是由一个格子一个格子构成,然后蛇在这个里面运动。
重写paint方法,单元格就是横着画几条线竖着画几条线即可。
代码如下:
@Override
public void paint(Graphics g) {
Color c = g.getColor();
g.setColor(Color.GRAY);
/*
* 将界面画成由ROW*COL的方格构成,两个for循环即可解决
* */
for(int i = 0;i<ROW;i++){
g.drawLine(0, i*BLOCK_HEIGHT, COL*BLOCK_WIDTH,i*BLOCK_HEIGHT );
}
for(int i=0;i<COL;i++){
g.drawLine(i*BLOCK_WIDTH, 0 , i*BLOCK_WIDTH ,ROW*BLOCK_HEIGHT);
}
g.setColor(c);
}
效果如下:
第三步完成的功能:建立另外的线程来控制重画
由于,蛇的运动就是改变蛇所在的位置,然后进行重画,就是我们所看到的运动。因此,在这里,我们单独用一个线程来控制重画。
1、新建一个MyPaintThread类,实现了Runnable接口
private class MyPaintThread implements Runnable{
@Override
public void run() {
//每隔50ms重画一次
while(true){
repaint();//会自动调用paint方法
try {
Thread.sleep(50);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}
2、在SnakeFrame的launchFrame方法中添加代码:new Thread(new MyPaintThread()).start();即可。
完成功能:利用双缓冲来解决闪烁的问题
private Image offScreenImage = null;
/*
* 重写update方法
* */
@Override
public void update(Graphics g) {
if(offScreenImage==null){
offScreenImage = this.createImage(ROW*BLOCK_HEIGHT, COL*BLOCK_WIDTH);
}
Graphics offg = offScreenImage.getGraphics();
//先将内容画在虚拟画布上
paint(offg);
//然后将虚拟画布上的内容一起画在画布上
g.drawImage(offScreenImage, 0, 0, null);
}
第四步完成的功能:在界面上画一个蛇出来
贪吃蛇游戏中的蛇就是用一系列的点来表示,这里我们来模拟一个链表。链表上的每个元素代表一个节点。
首先,我们先新建一个Node类来表示构成蛇的节点,用面向对象的思想,发现,这个类应该有如下的属性和方法:
1、位置
2、大小,即长度、宽度
3、方向
4、构造方法
5、draw方法
Node类的代码如下:
public class Node {
private static final int BLOCK_WIDTH = SnakeFrame.BLOCK_WIDTH;
private static final int BLOCK_HEIGHT = SnakeFrame.BLOCK_HEIGHT;
/*
* 每个节点的位置
* */
private int row;
private int col;
//方向
private Direction dir ;
private Node pre;
private Node next;
public Node(int row, int col, Direction dir) {
this.row = row;
this.col = col;
this.dir = dir;
}
public void draw(Graphics g){
Color c = g.getColor();
g.setColor(Color.BLACK);
g.fillRect(col*BLOCK_WIDTH, row*BLOCK_HEIGHT, BLOCK_WIDTH, BLOCK_HEIGHT);
g.setColor(c);
}
}
Direction是一个enum,具体如下:
public enum Direction {
L,U,R,D
}
而在Snake类中,用面向对象的思维,可以发现,Snake类中应该有如下的属性和方法
1、头结点
2、尾结点
3、构造函数
3、draw方法
具体代码如下:
public class Snake {
private Node head = null;
private Node tail = null;
private SnakeFrame sf;
//初始化是蛇的位置
private Node node = new Node(3,4,Direction.D);
private int size = 0;
public Snake(SnakeFrame sf) {
head = node;
tail = node;
size ++;
this.sf = sf ;
}
public void draw(Graphics g){
if(head==null){
return ;
}
for(Node node = head;node!=null;node = node.next){
node.draw(g);
}
}
}
在SnakeFrame类中new一个Snake对象,然后调用Snake对象的draw方法即可。
效果如下:
第五步完成的功能:通过键盘控制蛇的上下左右移动
首先想到的是这样:在Snake类中添加一个keyPressed方法,然后在SnakeFrame的键盘事件中调用Snake对象的keyPressed方法。
注意:蛇的移动是通过在头部添加一个单元格,在尾部删除一个单元格这样的思想来实现。
具体如下:
Snake类中添加一个keyPressed方法,主要是根据键盘的上下左右键来确定蛇的头结点的方向,然后move方法再根据头结点的方向来在头部添加一个单元格。
public void keyPressed(KeyEvent e) {
int key = e.getKeyCode();
switch(key){
case KeyEvent.VK_LEFT :
if(head.dir!=Direction.R){
head.dir = Direction.L;
}
break;
case KeyEvent.VK_UP :
if(head.dir!=Direction.D){
head.dir = Direction.U;
}
break;
case KeyEvent.VK_RIGHT :
if(head.dir!=Direction.L){
head.dir = Direction.R;
}
break;
case KeyEvent.VK_DOWN :
if(head.dir!=Direction.U){
head.dir = Direction.D;
}
break;
}
}
public void move() {
addNodeInHead();
deleteNodeInTail();
}
private void deleteNodeInTail() {
Node node = tail.pre;
tail = null;
node.next = null;
tail = node;
}
private void addNodeInHead() {
Node node = null;
switch(head.dir){
case L:
node = new Node(head.row,head.col-1,head.dir);
break;
case U:
node = new Node(head.row-1,head.col,head.dir);
break;
case R:
node = new Node(head.row,head.col+1,head.dir);
break;
case D:
node = new Node(head.row+1,head.col,head.dir);
break;
}
node.next = head;
head.pre = node;
head = node;
}
//最后,在draw中调用move方法即可
public void draw(Graphics g){
if(head==null){
return ;
}
move();
for(Node node = head;node!=null;node = node.next){
node.draw(g);
}
}
这样就实现了通过键盘来实现蛇的移动。
完成的功能:蛇吃蛋
首先我们新建一个蛋Egg的类。
类的属性和方法有:
1、位置、大小
2、构造方法
3、draw方法
4、getRect方法:用于碰撞检测
5、reAppear方法:用于重新产生蛋的方法
代码如下:
public class Egg {
//所在的位置
private int row;
private int col;
//大小
private static final int BLOCK_WIDTH = SnakeFrame.BLOCK_WIDTH;
private static final int BLOCK_HEIGHT = SnakeFrame.BLOCK_HEIGHT;
private static final Random r = new Random();
private Color color = Color.RED;
public Egg(int row, int col) {
this.row = row;
this.col = col;
}
public Egg() {
this((r.nextInt(SnakeFrame.ROW-2))+2,(r.nextInt(SnakeFrame.COL-2))+2);
}
/*
* 改变当前对象的位置,即完成蛋的重现
* */
public void reAppear(){
this.row = (r.nextInt(SnakeFrame.ROW-2))+2;
this.col = (r.nextInt(SnakeFrame.COL-2))+2;
}
public void draw(Graphics g){
Color c= g.getColor();
g.setColor(color);
g.fillOval(col*BLOCK_WIDTH, row*BLOCK_HEIGHT, BLOCK_WIDTH, BLOCK_HEIGHT);
g.setColor(c);
//改变下一次的颜色
if(color==Color.RED){
color = Color.BLUE;
}
else{
color = Color.RED;
}
}
//用于碰撞检测
public Rectangle getRect(){
return new Rectangle(col*BLOCK_WIDTH, row*BLOCK_HEIGHT, BLOCK_WIDTH, BLOCK_HEIGHT);
}
}
蛇吃蛋,怎么样才能判断蛇吃到蛋了呢,这就需要用到碰撞检测了。
这里我们在Snake类中添加一个eatEgg方法。当蛇吃到蛋之后,就需要将蛇的长度+1,这里处理的是在蛇的头部添加一个节点,当蛋被吃掉之后,就需要再重新随机产生一个蛋。
代码如下:
public Rectangle getRect(){
return new Rectangle(head.col*BLOCK_WIDTH, head.row*BLOCK_HEIGHT, BLOCK_WIDTH, BLOCK_HEIGHT);
}
public boolean eatEgg(Egg egg){
if(this.getRect().intersects(egg.getRect())){
addNodeInHead();
egg.reAppear();
return true;
}
else{
return false;
}
}
以上就完成了蛇吃蛋的功能。
完成的功能:添加边界处理
在我们熟悉的贪吃蛇游戏中,我们一般都知道,当蛇撞到墙或者是撞到自己身体的某一部分,则游戏就结束。下面我们就来实现这一功能。
在Snake类中,添加checkDead方法
private void checkDead() {
//头结点的边界检查
if(head.row<2||head.row>SnakeFrame.ROW||head.col<0||head.col>SnakeFrame.COL){
this.sf.gameOver();
}
//头结点与其它结点相撞也是死忙
for(Node node =head.next;node!=null;node = node.next){
if(head.row==node.row&&head.col == node.col){
this.sf.gameOver();
}
}
}
如果蛇撞墙或是撞到自己本身的某一个部分。则调用SnakeFrame类中的gameOver()方法来进行一定的处理。
本游戏的处理方法为:通过设置一个boolean 变量,来停止游戏并提示相关信息。
具体代码如下:
private boolean b_gameOver = false;
public void gameOver(){
b_gameOver = true;
}
@Override
public void update(Graphics g) {
//其它代码省略
if(b_gameOver){
g.drawString("游戏结束!!!", ROW/2*BLOCK_HEIGHT, COL/2*BLOCK_WIDTH);
}
}
以上就完成了蛇是否撞墙或是撞到自身一部分的功能。
小结
以上基本上实现了贪吃蛇的基本功能。剩下的一些功能不再介绍,例如:添加得分记录、通过键盘某按键来控制游戏的停止、重新开始、再来一局等。
以上的功能虽然没有介绍,但是在代码中,没有实现这些相应的功能。
完整代码可以在这里获取:
https://github.com/wojiushimogui/Snake
- 上一篇:金士顿Canvas Go!Plus microSD卡评测 4K影像轻松驾驭
- 已经是最后一篇了
相关推荐
- 如何用Java还原童年回忆?在线教你完成贪吃蛇小游戏
-
今天我就从零开始来完成这个小游戏,完成的方式也是一步一步的添加功能这样的方式来实现。额,不好意思,放错了,重来第一步完成的功能:写一个界面大家见到的贪吃蛇小游戏,界面肯定是少不了的。因此,第一步就是写...
- 金士顿Canvas Go!Plus microSD卡评测 4K影像轻松驾驭
-
【ZOL中关村在线原创评测】如今,视频已经成为社交媒体的主流内容传播形式,全民自媒体时代更是让昔日被视为高端、专业的视频拍摄,走进大众的生活。同时,无人机、运动相机等新世代影像设备也已经支持了4K视频...
- 国外顶尖教程大师教你设计一个炫酷游戏海报...
-
今天的教程很实用,我想对于做网页设计的童鞋肯定非常想学会用Photoshop的合成技巧来制作高端的游戏网站我们将用很多素材和照片来合成一个场景和一些页头和导航的制作。过程需要很长时间,素材也很多,我自...
- Excel 制作贪吃蛇游戏,让你轻松摸鱼!
-
步骤1:准备工作表1.创建游戏区域:o在单元格区域(如B2:AK30)设置一个矩形区域,调整行高和列宽为正方形(如行高20像素,列宽3字符)。o设置背景色为深色(如黑色),作为游戏画布。2.初...
- 成长的画布,绘满童趣色彩
-
在万家丽热闹的脉动中,特步童装是一方柔软的画布,等待孩子们用欢笑与奔跑泼洒斑斓色彩。这里没有生硬的商业气息,每一件衣物都像会呼吸的精灵,悄悄诉说着关于童年的奇妙物语。漫步其中,仿佛踏入一座流动的童话森...
- 【教育高质量发展】施甸县示范小学:地面为画布 游戏“绘”出精彩童年
-
近日,施甸县示范小学积极践行“健康第一”教育理念,在寒假前夕,学校美术组教师联合部分师生完成了校园地面游戏彩绘工作。教师们以地面为画布,绘制跳格子、转盘、迷宫等游戏图案,为下学期师生15分钟课间做好了...
- 《国王的画布》Steam试玩发布 手绘风地图制作工具
-
HannesBreuer工作室制作并发行,一款手绘风地图制作工具游戏《国王的画布》Steam试玩发布,本作暂不支持中文。《国王的画布》游戏中玩家可以自定义交互式路径生成随机元素,易于调整改动。无需...
- 怪物收集RPG冒险游戏《妖之乡》7月16日正式发售
-
今日(6月18日),怪物收集RPG冒险游戏《妖之乡》更新发售日消息,该作将于7月16日发售,游戏试玩Demo现已正式上线,感兴趣的玩家可以进入商店页面。游戏介绍:《妖之乡》是一款东方奇幻的怪物收集R...
- 208元起,育碧第一人称动作冒险游戏《阿凡达:潘多拉边境》发售
-
IT之家6月18日消息,育碧旗下第一人称动作冒险游戏《阿凡达:潘多拉边境(Avatar:FrontiersofPandora)》已于今天登陆Steam平台(点此访问),目前本作正在平台...
- 甜蜜下潜藏的疯狂?和病娇美少女《米塔》在一起的冒险解谜游戏今日发售
-
由Aihasto开发制作,IndieArk发行的冒险解谜游戏《米塔》在今日上架Steam平台进行发售。售价人民币52元,首发-10%的折扣持续14天,折扣后仅需46.8元。该作同步支持日语和俄语配音...
- 十大必玩的合作单机游戏排行
-
合作模式一直是单机游戏中的重要元素,玩家可以和朋友一起组队、配合,共同完成游戏任务。在众多的合作单机游戏中,哪些才是最值得一试的呢?本文将为大家介绍十大必玩的合作单机游戏排行。这些游戏不仅拥有精彩刺激...
- 原版20周年献礼:《战争机器:重装上阵》游戏8月26日发行
-
IT之家5月6日消息,微软官方XboxWire博客昨日(5月5日)发布博文,官宣《战争机器:重装上阵》(GearsofWar:Reloaded)游戏,将于2025年8...
- 十款免费的战斗单机游戏推荐之网络版
-
这篇文章主要介绍了十款免费的战斗单机游戏,并对它们进行了排行。这些游戏涵盖了不同类型和风格,包括角色扮演、射击、策略等。每款游戏都有详细的介绍和评价,读者可以根据自己的喜好选择适合自己的游戏。对于喜欢...
- 十大必玩的合作单机游戏:打造完美团队
-
合作模式一直是单机游戏中的重要元素,玩家可以通过合作来共同完成游戏任务,增加游戏的趣味性和挑战性。在本文中,我们将为您介绍十大必玩的合作单机游戏。这些游戏不仅拥有精彩刺激的剧情和画面,还提供了多种合作...
- 十款好玩的战斗单机游戏:哪个更好玩
-
战斗单机游戏一直是玩家们的最爱,而好玩的战斗单机游戏更是备受期待。在众多的战斗单机游戏中,哪些更好玩呢?本文将为您介绍十款备受好评的战斗单机游戏,并进行详细比较分析。无论您是喜欢动作还是策略,都能在这...
- 一周热门
- 最近发表
- 标签列表
-
- HTML 简介 (30)
- HTML 响应式设计 (31)
- HTML URL 编码 (32)
- HTML Web 服务器 (31)
- HTML 表单属性 (32)
- HTML 音频 (31)
- HTML5 支持 (33)
- HTML API (36)
- HTML 总结 (32)
- HTML 全局属性 (32)
- HTML 事件 (31)
- HTML 画布 (32)
- HTTP 方法 (30)
- 键盘快捷键 (30)
- CSS 语法 (35)
- CSS 选择器 (30)
- CSS 轮廓宽度 (31)
- CSS 谷歌字体 (33)
- CSS 链接 (31)
- CSS 定位 (31)
- CSS 图片库 (32)
- CSS 图像精灵 (31)
- SVG 文本 (32)
- 时钟启动 (33)
- HTML 游戏 (34)