最近JAVA和JSwing上手练习了一下贪吃蛇,供大家参考,具体内容如下

欢迎交流和加入新的内容

用到了JSwing,下面是一些具体的思路

实现

* 蛇:

采用单链表记录首尾,整个蛇被分为lattice格子,放在map里

* 移动:

我在实现的过程中发现最难得反而是蛇的定义和实现。一直想着怎么样用单独的方法表示出蛇来,但是如果将蛇单独实现,总有些细节实现起来特别麻烦

其实蛇移动并非牵一发而动全身,其实身子是没有发生变化的,关键是两点:

a.头的移动

b.尾巴的移动

实现:

直接把蛇实现在地图的小格子里,不再单独设置子类或者ArrayList等,Map里加上蛇头的坐标,从而使得Map可以根据蛇头改变蛇的坐标(类似于变量交换)。为头部单独设置x,y,作为移动的方向(也可以作为静态变量x和y,不过没什么区别),为身子设置next指针,只要next.next不是尾巴,那么保持不变。如果next是尾巴,就把自己的设置为尾巴,并且改变next,使之成为普通地图块。(refresh方法)

* 控制方向:

使用键盘事件,目前仅设置了wasd四个

* 窗口设计:

view extends JPanel,控制显示,然后在Lattice里调用Graphics.draw(...)实现对每个格子的显示

下面是核心的map部分代码(包括自动移动,检测食物,增加长度等等)

import codes.myGame.snake.cell.Lattice;

import java.util.Random;

public class Smap {

private boolean getFood = false;//如果得到食物,该指针设为true,并且在随后的autoChange里增加蛇的长度

private boolean gameOver = false;

private boolean directionChange = false;//这里标志的作用是保证在一次运动期间只会进行一次转向,使游戏更流畅

private int MAP_SIZE;

private Lattice[][] map;

private int directionX = 0;//下一次头在当前位置的哪个方向上

private int directionY = 1;//下一次头在当前位置的哪个方向上

private int[] head = new int[2];//记录当前头的位置

private int[] food = new int[2];//记录当前食物的位置

public Smap(int size) {

MAP_SIZE = size;

map = new Lattice[MAP_SIZE][MAP_SIZE];

for(int i=0;i

for (int j = 0 ;j

map[i][j] = new Lattice();

}

}

map[MAP_SIZE/2][MAP_SIZE/2].setHead(true,map[MAP_SIZE/2][MAP_SIZE/2-1]);//初始化设置一个头结点,以及他的尾节点

head[0] = MAP_SIZE/2;

head[1] = MAP_SIZE/2;

map[MAP_SIZE/2][MAP_SIZE/2-1].setRear(true,null);

this.randFood();

}

//模拟蛇的自动移动

public void autoChange(){

this.setHead();

if(food[0]==head[0] && food[1]==head[1]){//如果新的头部碰触到了食物,那么尾部增长

getFood = true;

}

if(!gameOver)this.setRear();

if(getFood)this.randFood();

directionChange = false;

}

//根据键盘事件,改变头的下一次移动方向,注意 该移动方向是仅针对头部的

//setDirection和setHead两个方法需要互斥进行,这里单线程,用synchronized即可

//(否则,如果当前头部在边界位置,连续变幻方向可能导致在setHead里发生溢出)

public synchronized void setDirection(int x,int y){

if(directionY!=y && directionX!=x &&!directionChange){

directionX = x;

directionY = y;

directionChange = true;

}

}

public boolean gameOver(){

return gameOver;//头碰到身子,证明gameOver

}

private synchronized void setHead(){

int i = head[0];

int j = head[1];

head[0] = ( head[0] + directionX + MAP_SIZE)%MAP_SIZE;

head[1] = ( head[1] + directionY + MAP_SIZE )%MAP_SIZE;

if(map[head[0]][head[1]].isBody())gameOver = true;

map[head[0]][head[1]].setHead(true,map[i][j]);

map[i][j].setBody(true,null);

map[i][j].setHead(false,null); //传入null表示不改变当前指向

}

//设置尾巴由于没法像头部那样直接设置,这里只能采用链表遍历的方式获取尾巴

private void setRear(){

if(!getFood){

Lattice temp = map[head[0]][head[1]];

while (!temp.next.isRear())temp = temp.next;

temp.next().setRear(false,null);

temp.setRear(true,null);

temp.setBody(false,null);

}

}

private void randFood(){

getFood = false;

map[food[0]][food[1]].setFood(false);//先把当前的食物取消掉

boolean flag = false;//设置下一个食物

Random random = new Random();

int x = random.nextInt(MAP_SIZE);

int y = random.nextInt(MAP_SIZE);

while (!flag){

x = random.nextInt(MAP_SIZE);

y = random.nextInt(MAP_SIZE);

if(!map[x][y].isHead() && !map[x][y].isRear() &&!map[x][y].isBody())flag = true;

}

map[x][y].setFood(true);

food[0] = x;

food[1] = y;

}

public Lattice get(int row, int col){

return map[row][col];

}

public int getMAP_SIZE() {

return MAP_SIZE;

}

}

下面是显示部分的代码

显示分为两部分,一块是利用Graphics.draw()方法实现单个单元格的绘制,另一块设置view类继承自JPanel。负责绘制图画显示

public class Lattice {

private boolean isBody = false;

private boolean isHead = false;

private boolean isFood = false;

private boolean isRear = false;

public Lattice next = null;

public void setHead(boolean bool,Lattice next){

isHead = bool;

if(next!=null)this.next = next;

}

public void setBody(boolean bool,Lattice next){

isBody = bool;

if(next!=null)this.next = next; //传入参数为null时,不改变当前的next

}

public void setRear(boolean bool,Lattice next){

isRear = bool;

this.next = next;

}

public void setFood(boolean bool){

isFood = bool;

}

public Lattice next(){

return next;

}

public boolean isHead(){

return isHead;

}

public boolean isFood(){

return isFood;

}

public boolean isRear(){

return isRear;

}

public boolean isBody(){

return isBody;

}

public void refresh(){

if(isHead){

isBody = true;

isHead = false;

// 怎么设置下一个头呢?(考虑把DirectionX,Y放到Smap里,而不是这里)

}else if(isBody){

if(next.isRear){

next.isRear = false;

isRear = true;

isBody = false;

}

}

}

// 在这里设置细胞可见

public void draw(Graphics g, int x, int y, int size) {

g.setColor(black);

g.drawRect(x, y, size, size);

if ( isHead ) {

g.setColor( red);

g.fillRect(x, y, size, size);

}else if ( isBody || isRear) {

g.setColor(black);

g.fillRect(x, y, size, size);

}else if(isFood){

g.setColor( blue);

g.fillRect(x, y, size, size);

}

}

}

view部分:

import codes.myGame.snake.cell.Lattice;

import javax.swing.*;

import java.awt.*;

public class View extends JPanel {

private static final long serialVersionUID = -5258995676212660595L;

private static final int GRID_SIZE = 32; //填充的像素数量

private Smap thisMap;

public View(Smap map) {

thisMap = map;

}

@Override

public void paint(Graphics g) {

super.paint(g);

int size = thisMap.getMAP_SIZE();

for (int row = 0; row< size; row++ ) {

for (int col = 0; col< size; col++ ) {

Lattice lattice = thisMap.get(row, col);

if ( lattice != null ) {

lattice.draw(g, col*GRID_SIZE, row*GRID_SIZE, GRID_SIZE);//对应的格子的显示

}

}

}

}

@Override

public Dimension getPreferredSize() {//创建该div大小

return new Dimension(thisMap.getMAP_SIZE()*GRID_SIZE+1, thisMap.getMAP_SIZE()*GRID_SIZE+1);

}

}

更多有趣的经典小游戏实现专题,分享给大家:

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

贪吃蛇游戏java代码_Java实现贪吃蛇游戏相关推荐

  1. 扑克牌游戏java代码_java实现简易扑克牌游戏

    本文实例为大家分享了java实现扑克牌游戏的具体代码,供大家参考,具体内容如下 游戏功能描述 1.创建一副扑克牌: 包括四种花色:黑桃,红桃,梅花,方片 包括十三种点数:2-10,J,Q,K,A 2. ...

  2. 拼图游戏java代码_java设计:拼图游戏 完整源码

    //July/设计. //我对有的老师,很无语./06.04/. //Puzzle.java ----------------------------------------------------- ...

  3. 贪吃蛇java代码_java实现贪吃蛇的代码实例

    用几天的空闲时间写个贪吃蛇. 下面的代码都打了注释,如果有什么问题或者有什么指点的地方希望留言不吝赐教! 下载地址(需要1积分,如果不想用积分,直接拷贝下面的代码即可) Board类:import j ...

  4. 2048 java代码_java实现的2048游戏完整实例

    本文实例讲述了java实现的2048游戏.分享给大家供大家参考,具体如下: 先来看看运行效果: 具体代码如下: package awtDemo; import javax.swing.*; impor ...

  5. 蜘蛛纸牌 java代码_java实现蜘蛛纸牌游戏

    [实例简介] 用java技术实现windows自带的蜘蛛纸牌游戏 [实例截图] [核心代码] CardsGame └── CardsGame ├── bin │   └── com │   └── g ...

  6. 小学数学闯关游戏 java代码_Java语言实现小学数学练习

    package per.java.shejiti; import java.io.*; import java.util.ArrayList; import java.util.Collections ...

  7. 拼图java代码_Java制作智能拼图游戏原理及代码

    今天突发奇想,想做一个智能拼图游戏来给哄女友. 需要实现这些功能 第一图片自定义 第二宫格自定义,当然我一开始就想的是3*3 4*4 5*5,没有使用3*5这样的宫格. 第三要实现自动拼图的功能,相信 ...

  8. 弹球小游戏java代码_java实现简单的弹球游戏

    弹球游戏实现原理: 隔一定时间(小于1秒)重新绘制图像,因为Graphics类是一个抽象类,创建子类的时候需要把所有涉及的方法都得重写,所以这里使用的是创建Canvas的子类,只需要重写它的paint ...

  9. 弹球小游戏java代码_android 发射弹球小游戏 附完整源码下载

    [实例简介]android 弹球游戏制作 [实例截图]  [核心代码] package wyf.ytl; //声明包语句 public class BallGoThread extends Threa ...

最新文章

  1. MMX Intrinsics各函数介绍
  2. jQueryDOM操作
  3. js读取解析JSON类型数据【申明:来源于网络】
  4. 可持久化Splay 学习笔记
  5. python根据文件名列表筛选满足条件的文件
  6. 主流区块链底链技术横评 hyperledger fabric、fisco bcos、chainmaker
  7. python 连接数据库 慢_python自动结束mysql慢查询会话的实例代码
  8. matlab画中国,用matlab绘制中国地图
  9. 跟进分析不一样的MTI商城
  10. 2021年MEMS传感器产业链变化与趋势
  11. 为Exynos4412移植2022版U-Boot(一)步骤及其原理分析
  12. 解决Microsoft Edge与Chrome地址栏使用必应搜索bing.com很慢的问题
  13. 虚拟化kvm-虚拟化概述
  14. 动视是否磨灭了暴雪的灵魂?
  15. 手游开发神器 cocos2d-x editor 教程聚合和代码下载(持续更新中)
  16. 【BUCT数据结构类库】1.2--链表的基本操作
  17. 【网络】内网访问外网和外网访问内网的原理
  18. 原来开车和写博客是很相似的心理变化过程
  19. 影院活动管理系统 项目设计阶段
  20. 一张图看懂微信小程序全生态!

热门文章

  1. ZooKeeper分布式应用程序的分布式协调服务:概述,入门,发布版本
  2. php output详解,PHP输出缓冲控制Output Control系列函数详解,output函数详解
  3. c++软件开发面试旋极面试题_腾讯软件开发面试题(有详细解答)
  4. Page 和Ability之间区别
  5. web前端基础(08html5新标签)
  6. python聚类分析如何确定分类个数_R语言做聚类分析Kmeans时确定类的个数
  7. 接二手java项目需要什么资料_师妹问双非本科如何进BAT,我告诉她Java后端路线...
  8. 西安科技大学计算机学院保研,独臂姑娘,好样的!
  9. java中fmt标签库_jsp fmt标签详解
  10. OpenShift 4 Tekton (1) - OpenShift Pipeline入门-安装Pipeline Operator