java中使用双向链表实现贪吃蛇程序源码分享

使用双向链表实现贪吃蛇程序

1.链表节点定义:

package snake;

public class SnakeNode {
	private int x;
	private int y;
	private SnakeNode next;
	private SnakeNode ahead;

	public SnakeNode() {
	}

	public SnakeNode(int x, int y) {
		super();
		this.x = x;
		this.y = y;
	}

	public int getX() {
		return x;
	}

	public void setX(int x) {
		this.x = x;
	}

	public int getY() {
		return y;
	}

	public void setY(int y) {
		this.y = y;
	}

	public SnakeNode getNext() {
		return next;
	}

	public void setNext(SnakeNode next) {
		this.next = next;
	}

	public SnakeNode getAhead() {
		return ahead;
	}

	public void setAhead(SnakeNode ahead) {
		this.ahead = ahead;
	}

}

主程序:

package snake;

import java.awt.BorderLayout;
import java.awt.GridLayout;
import java.awt.KeyEventPostProcessor;
import java.awt.KeyboardFocusManager;
import java.awt.event.KeyEvent;
import java.util.Random;

import javax.swing.ImageIcon;
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JOptionPane;
import javax.swing.JPanel;

/**
 * Created by hackcoder on 15-3-11.
 */
public class Snake extends JFrame {
	private static final int rows = 60;
	private static final int columns = 80;

	// 方向
	private static final int UP = 1;
	private static final int RIGHT = 2;
	private static final int DOWN = 3;
	private static final int LEFT = 4;

	private static int DRIECTION_NOW = RIGHT;
	private static boolean isEat = false;

	private static int TAILX;
	private static int TAILY;

	private static SnakeNode snakeHeader = new SnakeNode();
	private static SnakeNode snakeTailer = snakeHeader;
	private static SnakeNode food = new SnakeNode();
	private static JLabel[] images = new JLabel[rows * columns];

	public static void main(String args[]) {
		snakeHeader.setX(new Random().nextInt(rows - 1));
		snakeHeader.setY(new Random().nextInt(columns - 1));
		Snake snake = new Snake();
		food = getFood();
		while (true) {
			try {
				next();
				// 吃到了食物
				if (food.getX() == snakeHeader.getX()
						&& food.getY() == snakeHeader.getY()) {
					addTail();
					isEat = true;
				}
				//吃到食物,重新生成一个食物
				if (isEat) {
					food = getFood();
				}
				// 判断是否结束游戏
				if (judgeEND()) {
					JOptionPane.showMessageDialog(null, "游戏结束!", "游戏结束!",
							JOptionPane.ERROR_MESSAGE);
					break;
				}
				SnakeNode pNow = snakeHeader;
				while (pNow != null) {
					images[columns * pNow.getX() + pNow.getY()]
							.setIcon(new ImageIcon("image/black.jpg", ""));
					pNow = pNow.getNext();
				}
				images[columns * food.getX() + food.getY()]
						.setIcon(new ImageIcon("image/black.jpg", ""));
				Thread.sleep(100);
				// 清理
				pNow = snakeHeader;
				while (pNow != null) {
					images[columns * pNow.getX() + pNow.getY()]
							.setIcon(new ImageIcon("image/white.jpg", ""));
					pNow = pNow.getNext();
				}
				images[columns * food.getX() + food.getY()]
						.setIcon(new ImageIcon("image/white.jpg", ""));

				isEat = false;
			} catch (InterruptedException e) {
				e.printStackTrace();
			}
		}
	}

	public Snake() {
		init();
		this.setBounds(80, 80, 400, 400);
		this.setVisible(true);
		setDefaultCloseOperation(javax.swing.WindowConstants.EXIT_ON_CLOSE);
		// 添加全局键盘监听
		KeyboardFocusManager manager = KeyboardFocusManager
				.getCurrentKeyboardFocusManager();
		manager.addKeyEventPostProcessor((KeyEventPostProcessor) this
				.getMyKeyEventHandler());
	}

	/**
	 * 初始化地图
	 */
	public void init() {
		JPanel p = new JPanel(new GridLayout(rows, columns, 1, 1));
		setLayout(new BorderLayout());
		for (int x = 0; x < rows; x++) {
			for (int y = 0; y < columns; y++) {
				ImageIcon imageIcon;
				if (x == 0 || x == rows - 1 || y == 0 || y == columns - 1) {
					imageIcon = new ImageIcon("image/red.jpg", "");
				} else {
					imageIcon = new ImageIcon("image/white.jpg", "");
				}
				images[columns * x + y] = new JLabel(imageIcon);
				p.add(images[columns * x + y]);
			}
		}
		getContentPane().add(p, BorderLayout.CENTER);

	}

	/**
	 * 键盘监听
	 *
	 * @return
	 */
	public KeyEventPostProcessor getMyKeyEventHandler() {
		return new KeyEventPostProcessor() {
			public boolean postProcessKeyEvent(KeyEvent e) {
				if (e.getID() != KeyEvent.KEY_PRESSED) {
					return false;
				}
				int keycode = e.getKeyCode();
				if (keycode == KeyEvent.VK_UP) {

					if (snakeHeader.getNext() != null) {
						// 判断方向是否可转
						int x1 = snakeHeader.getX();
						int y1 = snakeHeader.getY();
						int x2 = snakeHeader.getNext().getX();
						int y2 = snakeHeader.getNext().getY();
						if (y1 == y2 && x1 - x2 == 1) {
							return true;
						}
					}
					DRIECTION_NOW = UP;
				} else if (keycode == KeyEvent.VK_RIGHT) {
					if (snakeHeader.getNext() != null) {
						int x1 = snakeHeader.getX();
						int y1 = snakeHeader.getY();
						int x2 = snakeHeader.getNext().getX();
						int y2 = snakeHeader.getNext().getY();
						if (x1 == x2 && y2 - y1 == 1) {
							return true;
						}
					}
					DRIECTION_NOW = RIGHT;
				} else if (keycode == KeyEvent.VK_DOWN) {
					if (snakeHeader.getNext() != null) {
						int x1 = snakeHeader.getX();
						int y1 = snakeHeader.getY();
						int x2 = snakeHeader.getNext().getX();
						int y2 = snakeHeader.getNext().getY();
						if (y1 == y2 && x2 - x1 == 1) {
							return true;
						}
					}
					DRIECTION_NOW = DOWN;
				} else if (keycode == KeyEvent.VK_LEFT) {
					if (snakeHeader.getNext() != null) {
						int x1 = snakeHeader.getX();
						int y1 = snakeHeader.getY();
						int x2 = snakeHeader.getNext().getX();
						int y2 = snakeHeader.getNext().getY();
						if (x1 == x2 && y1 - y2 == 1) {
							return true;
						}
					}
					DRIECTION_NOW = LEFT;
				}
				return true;
			}
		};
	}

	/**
	 * 计算贪吃蛇的方向及位移
	 *
	 * @param header
	 */
	public static void next() {
		if (snakeHeader == null)
			return;
		TAILX = snakeTailer.getX();
		TAILY = snakeTailer.getY();
		SnakeNode pNow = snakeTailer;
		while (pNow != null) {
			if (pNow == snakeHeader) {
				break;
			}
			pNow.setX(pNow.getAhead().getX());
			pNow.setY(pNow.getAhead().getY());
			pNow = pNow.getAhead();
		}

		if (DRIECTION_NOW == RIGHT) {
			snakeHeader.setY(snakeHeader.getY() + 1);
		} else if (DRIECTION_NOW == LEFT) {
			snakeHeader.setY(snakeHeader.getY() - 1);
		} else if (DRIECTION_NOW == UP) {
			snakeHeader.setX(snakeHeader.getX() - 1);
		} else if (DRIECTION_NOW == DOWN) {
			snakeHeader.setX(snakeHeader.getX() + 1);
		}
	}

	public static void addTail() {
		SnakeNode tail = new SnakeNode(TAILX, TAILY);
		snakeTailer.setNext(tail);
		tail.setAhead(snakeTailer);
		snakeTailer = snakeTailer.getNext();

	}

	public static SnakeNode getFood() {
		SnakeNode food = new SnakeNode();
		boolean flag = true;
		while (true) {
			int x = new Random().nextInt(rows);
			int y = new Random().nextInt(columns);
			if (x == 0 || x == rows - 1 || y == 0 || y == columns - 1) {
				continue;
			}
			SnakeNode pNow = snakeHeader;
			while (pNow != null) {
				if (x == pNow.getX() && y == pNow.getY()) {
					flag = false;
				}
				pNow = pNow.getNext();
			}
			if (flag) {
				food = new SnakeNode(x, y);
				break;
			}
		}
		return food;
	}

	public static boolean judgeEND() {
		//碰墙判断
		if (snakeHeader.getX() == 0 || snakeHeader.getX() == rows - 1
				|| snakeHeader.getY() == 0 || snakeHeader.getY() == columns - 1) {
			return true;
		}
		//碰身体判断
		SnakeNode pNow = snakeHeader.getNext();
		while (pNow != null) {
			if (snakeHeader.getX() == pNow.getX()
					&& snakeHeader.getY() == pNow.getY()) {
				System.out.println("=========碰到身体===========");
				return true;
			}
			pNow = pNow.getNext();
		}
		return false;
	}

}
(0)

相关推荐

  • Java语言中链表和双向链表

    链表是一种重要的数据结构,在程序设计中占有很重要的地位.C语言和C++语言中是用指针来实现链表结构的,由于Java语言不提供指针,所以有人认为在Java语言中不能实现链表,其实不然,Java语言比C和C++更容易实现链表结构.Java语言中的对象引用实际上是一个指针(本文中的指针均为概念上的意义,而非语言提供的数据类型),所以我们可以编写这样的类来实现链表中的结点. class Node { Object data; Node next;//指向下一个结点 } 将数据域定义成Object类是因为

  • java数据结构之实现双向链表的示例

    复制代码 代码如下: /** * 双向链表的实现 * @author Skip * @version 1.0 */public class DoubleNodeList<T> { //节点类 private static class Node<T>{  Node<T> perv;  //前节点  Node<T> next;  //后节点  T data;    //数据 public Node(T t){   this.data = t;  } } priv

  • java基于双向环形链表解决丢手帕问题的方法示例

    本文实例讲述了java基于双向环形链表解决丢手帕问题的方法.分享给大家供大家参考,具体如下: 问题:设编号为1.2--n的几个小孩围坐一圈,约定编号为k(1=<k<=n)的小孩从1开始报数,数到m的那个出列,他的下一位又从1开始报数,数到m的那个人又出列,直到所有人出列为止,由此产生一个出队编号的序列. 我们现在用一个双向环形链表来解这一问题.先来看看下面这幅图: 圆圈代表一个结点,红色的指针指向下一个元素,紫色的指针指向上一个元素.first指针指向第一个元素,表明第一个元素的位置,curs

  • Java数据结构及算法实例:汉诺塔问题 Hanoi

    /** * 汉诺塔大学的时候就学过,但是根本没搞明白,唯一知道的就是要用递归的方法来求解. * 问题描述: * 有三根杆子A,B,C.A杆上有N个(N>1)穿孔圆盘,盘的尺寸由下到上依次变小. * 要求按下列规则将所有圆盘移至C杆: * 1.每次只能移动一个圆盘: * 2.大盘不能叠在小盘上面. * 提示:可将圆盘临时置于B杆,也可将从A杆移出的圆盘重新移回A杆, * 但都必须尊循上述两条规则. * 问:如何移?最少要移动多少次? * 解决方法: * 假设只有2个盘子,柱子分别是A, B, C柱

  • Java递归算法经典实例(经典兔子问题)

    题目:古典问题:3个月起每个月都生一对兔子,小兔子长到第三个月后每个月又生一对兔子,假如兔子都不死,问每个月的兔子总数为多少? 分析:首先我们要明白题目的意思指的是每个月的兔子总对数:假设将兔子分为小中大三种,兔子从出生后三个月后每个月就会生出一对兔子, 那么我们假定第一个月的兔子为小兔子,第二个月为中兔子,第三个月之后就为大兔子,那么第一个月分别有1.0.0,第二个月分别为0.1.0, 第三个月分别为1.0.1,第四个月分别为,1.1.1,第五个月分别为2.1.2,第六个月分别为3.2.3,第

  • java实现单链表、双向链表

    本文实例为大家分享了java实现单链表.双向链表的相关代码,供大家参考,具体内容如下 java实现单链表: package code; class Node { Node next; int data; public Node(int data) { this.data=data; } } class LinkList { Node first; //头部 public LinkList() { this.first=null; } public void addNode(Node no) {

  • Java采用循环链表结构求解约瑟夫问题

    本文实例讲述了Java采用循环链表结构求解约瑟夫问题的方法.分享给大家供大家参考.具体分析如下: 这是第一次java考试的试题,对于没看过链表的同学来说就不会做,现在回头看看,还真不难. 约瑟夫问题: 有n个人,其编号分别为1,2,3,-,n.这n个人按顺序排成一个圈.现在给定s和d,从第s个人开始从1依次报数,数到d的人出列,然后又从下一个人开始又从1开始依次报数,数到d的人又出列,如此循环,直到最后所有人出列为止.要求定义一个节点类,采用循环链表结构求解约瑟夫问题. 以下java版的答案:

  • java双向循环链表的实现代码

    例1: 复制代码 代码如下: package com.xlst.util; import java.util.HashMap;import java.util.Map;import java.util.UUID; /*** 双向循环链表* 完成时间:2012.9.28* 版本1.0* @author xlst**/public class BothwayLoopLinked {/*** 存放链表长度的 map* * 如果简单使用 static int 型的 size 基本类型变量,则只能维护一个

  • 浅谈java实现背包算法(0-1背包问题)

    0-1背包的问题 背包问题(Knapsack problem)是一种组合优化的NP完全问题.问题可以描述为:给定一组物品,每种物品都有自己的重量和价格,在限定的总重量内,我们如何选择,才能使得物品的总价格最高.问题的名称来源于如何选择最合适的物品放置于给定背包中. 这是最基础的背包问题,特点是:每种物品仅有一件,可以选择放或不放. 用子问题定义状态:即f[i][v]表示前i件物品恰放入一个容量为v的背包可以获得的最大价值.则其状态转移方程便是: f[i][v]=max{ f[i-1][v], f

  • Java编程删除链表中重复的节点问题解决思路及源码分享

    一. 题目 在一个排序的链表中,存在重复的结点,请删除该链表中重复的结点,重复的结点不保留,返回链表头指针. 二. 例子 输入链表:1->2->3->3->4->4->5 处理后为:1->2->5 三. 思路 个人感觉这题关键是注意指针的指向,可以定义一个first对象(值为-1,主要用于返回操作后的链表),first.next指向head,定义一个last同样指向first(主要用于操作记录要删除节点的前一个节点),定义一个p指向head,指向当前节点.

  • Java基于循环递归回溯实现八皇后问题算法示例

    本文实例讲述了Java基于循环递归回溯实现八皇后问题.分享给大家供大家参考,具体如下: 运行效果图如下: 棋盘接口 /** * 棋盘接口 * @author Administrator * */ public interface Piece { abstract boolean isRow(int line); abstract boolean isCol(int line,int col); } 棋盘类: /** * 棋盘 * @author Administrator * */ public

随机推荐