开发经验谈:贪吃蛇游戏的MIDP实现核心

2016-02-19 13:40 5 1 收藏

get新技能是需要付出行动的,即使看得再多也还是要动手试一试。今天图老师小编跟大家分享的是开发经验谈:贪吃蛇游戏的MIDP实现核心,一起来学习了解下吧!

【 tulaoshi.com - 编程语言 】

相信大家都玩过Nokia手机上的贪吃蛇游戏。在该游戏中,玩家操纵一条贪吃的蛇在迷宫里行走,贪吃蛇按玩家所按的方向键折行,蛇头吃到各种食物(比如大力丸)后,会有各种反应(比如蛇身变长),假如贪吃蛇碰上墙壁或者自身的话,就GameOver了(当然也可能是减去一条生命)。
  
  要实现该游戏其实并不麻烦,要害就是要找到一个合适的核心算法。本文就给出一个参考实现,你可以基于该demo做扩展。要说明的一点是:本文只演示最核心的算法,要实现一个完整的游戏,你还需要做很多的扩展,重构。
  
  实例代码
  
  该程序包括3个Java文件。一个是SnakeMIDlet,另外2个分别是一个Canvas(SnakeCanvas)和一个代表贪吃蛇的类Snake: SnakeMIDlet.java
  import javax.microedition.lcdui.Display;
  import javax.microedition.midlet.MIDlet;
  import javax.microedition.midlet.
  MIDletStateChangeException;
  
  /**
  * @author Jagie
  */
  public class SnakeMIDlet extends MIDlet
  {
  
  protected void startApp()
  throws MIDletStateChangeException
  {
  // TODO Auto-generated method stub
  Display.getDisplay(this)
  .setCurrent(new SnakeCanvas());
  
  }
  
  /* (non-Javadoc)
  * @see javax.microedition
  .midlet.MIDlet#pauseApp()
  */
  protected void pauseApp()
  {
  // TODO Auto-generated method stub
  
  }
  
  /* (non-Javadoc)
  * @see javax.microedition
  .midlet.MIDlet#destroyApp(boolean)
  */
  protected void destroyApp(boolean arg0)
  throws MIDletStateChangeException
  {
  // TODO Auto-generated method stub
  
  }
  
  }
  
  SnakeCanvas.java
  
  import javax.microedition.lcdui.Canvas;
  import javax.microedition.lcdui.Graphics;
  
  /**
  * @author Jagie
  *
  */
  public class SnakeCanvas extends
  Canvas implements Runnable
  {
  
  Snake snake=new Snake();
  SnakeCanvas(){
  snake.init();
  new Thread(this).start();
  }
  
  protected void paint(Graphics g)
  {
  
  g.setColor(0);
  g.fillRect(0,0,this.getWidth(),
  this.getHeight());
  
  snake.paint(g);
  
  }
  
  /**
  * 游戏主线程,驱动蛇移动
  */
  
  public void run()
  {
  while(true){
  
  snake.move();
  repaint();
  
  try
  {
  Thread.sleep(50);
  } catch (InterruptedException e)
  {
  // TODO Auto-generated
  catch block
  e.printStackTrace();
  }
  }
  }
  
  /**
  * 按键相应,产生新蛇头
  */
  protected void keyPressed(int c)
  {
  int ga=this.getGameAction(c);
  
  switch (ga)
  {
  case Canvas.UP:
  snake.breakInto(1);
  break;
  
  case Canvas.DOWN:
  snake.breakInto(3);
  break;
  case Canvas.LEFT:
  snake.breakInto(4);
  break;
  case Canvas.RIGHT:
  snake.breakInto(2);
  break;
  }
  }
  }
  
  Snake.java
  
  import java.util.Vector;
  
  import javax.microedition.lcdui.Graphics;
  
  /**
  *
  * @author Jagie
  * 贪吃蛇
  */
  public class Snake
  {
  //蛇环节,每个环节为一个int[] sec
  //sec[0]:环节起点x,sec[1]:
  环节起点y,sec[2]:环节方向,sec[3]:
  环节长度
  Vector sections = new Vector();
  
  /**
  * 初始化sections
  * 开始的时候,整条蛇只有一段。
  *
  */
  public void init()
  {
  int[] head =
  { 10, 10, 2, 50 };
  sections.addElement(head);
  }
  
  /**
  * 绘制
  * @param g
  */
  public synchronized
  void paint(Graphics g)
  {
  if (sections.isEmpty())
  {
  return;
  }
  g.setColor(0, 255, 0);
  for (int i = 0; i sections.size();
  i++)
  {
  int[] sec = (int[])
  sections.elementAt(i);
  //sec[0]:起点x,sec[1]:
  起点y,sec[2]:方向,sec[3]:
  长度
  switch (sec[2]) {
  case 1:
  g.drawLine(sec[0], sec[1],
  sec[0], sec[1] - sec[3]);
  break;
  case 2:
  g.drawLine(sec[0], sec[1],
  sec[0] + sec[3], sec[1]);
  break;
  case 3:
  g.drawLine(sec[0], sec[1],
  sec[0], sec[1] + sec[3]);
  break;
  case 4:
  g.drawLine(sec[0], sec[1],
  sec[0] - sec[3], sec[1]);
  break;
  }
  
  }
  }
  
  /**
  *
  * @author Jagie
  *
  * 蛇的爬行。本质上是蛇头长度++,蛇尾长度--。
  同时移动蛇尾起点。假如蛇尾长度小于0,则去掉蛇尾。
  */
  public synchronized void move()
  {
  if (sections.isEmpty())
  {
  return;
  }
  //蛇尾
  int[] tail = (int[])
  sections.elementAt
  (sections.size() - 1);
  //蛇头
  int[] head = (int[])
  sections.elementAt(0);
  //根据蛇尾环节的方向移动蛇尾。
  switch (tail[2])
  {
  case 1:
  tail[1]--;
  break;
  case 2:
  tail[0]++;
  break;
  case 3:
  tail[1]++;
  break;
  case 4:
  tail[0]--;
  break;
  }
  //蛇尾缩短
  tail[3]--;
  //蛇头增长
  head[3]++;
  //蛇尾0,则去掉蛇尾
  if (tail[3] = 0)
  {
  sections.removeElement(tail);
  }
  }
  
  /**
  * 蛇分段
  * @param dir 新蛇头的方向
  */
  
  public synchronized void
  breakInto(int dir)
  {
  if (sections.isEmpty())
  {
  return;
  }
  int[] head = (int[])
  sections.elementAt(0);
  //新蛇头方向和旧蛇头方向一致,
  则无反应。
  //TODO 可以考虑加速。
  if (dir == head[2])
  {
  return;
  }
  //增加新蛇头
  int[] newhead=new int[4];
  //新蛇头的起始位置,
  与旧蛇头的运动方向有关。
  switch (head[2])
  {
  case 1:
  newhead[0]=head[0];
  newhead[1]=head[1]-head[3];
  newhead[2]=dir;
  newhead[3]=0;
  //蛇头总是第一个元素
  sections.insertElementAt(newhead, 0);
  break;
  case 2:
  newhead[0]=head[0]+head[3];
  newhead[1]=head[1];
  newhead[2]=dir;
  newhead[3]=0;
  sections.insertElementAt(newhead, 0);
  break;
  case 3:
  newhead[0]=head[0];
  newhead[1]=head[1]+head[3];
  newhead[2]=dir;
  newhead[3]=0;
  sections.insertElementAt(newhead, 0);
  break;
  case 4:
  newhead[0]=head[0]-head[3];
  newhead[1]=head[1];
  newhead[2]=dir;
  newhead[3]=0;
  sections.insertElementAt(newhead, 0);
  break;
  }
  
  }
  
  }

来源:http://www.tulaoshi.com/n/20160219/1604943.html

延伸阅读
职场妈妈母乳喂养经验谈 许多妈妈在宝宝4个月或6个月以后,产假期满就得回单位上班了。这时妈妈就不便按时给宝宝哺乳了,需要进行混合喂养。而此时宝宝正需要添加辅食,如果喂养不当,很容易引起营养不良。同时,这个时期宝宝体内从母体中带来的一些免疫物质正在不断消耗、减少,若过早中断母乳喂养会导致抵抗力下降、消化功能紊乱,影响...
标签: 生活常识
验楼经验谈 图老师阅读配图   一、土建工程的验收楼宇整体外观不出现倾斜, 房间内的梁、柱、墙无明显的尺寸偏差(请带上钢卷尺);墙面、楼板面、天花等不出现明显的裂缝;批荡层不允许有龟裂缝、脱皮、起沙眼以及起泡等缺陷;门窗与墙身之间结合部的处理应平整,无缺棱掉角。 二、给排水工程的验收 1、给排水管之间的...
标签: Web开发
一、垃圾还是经典 网页技术更新很快,一个网站的界面设计寿命仅仅2-3年而已。不管是垃圾还是精品,都没有所谓的经典。经典只存在于是哪个首次成功创新性的应用。网页设计者不管自己的学识、技术和经验如何,都自以为自己吊的不得了,这可能是源与商业设计自我意识体现的强烈主观因素。一个闭门造车者做出的东西,是远远赶不上综合借鉴者的。...
摄影初学经验谈   对于初学者来说,摄影入门并不只是使用单反拍照那么简单,抛开各种复杂的参数不说,从选取题材、器材使用到拍摄技巧都需要仔细研究。为了大家在拍照时少走弯路,小编结合日常拍摄经验,为大家总结出了7条拍摄建议,一起来看一下吧。 1. 养成备份、清空储存卡的好习惯 相信很多人都有类似不良习惯,...
标签: Web开发
背景!jsp+mysql 记住 要用mysql的longblob类型来存默认的blob大小不够 数据库字段:id (char) pic (longblob) 转载请注明出处,这时我与我的知己的合作的结过 原来操作blob字段时都要先差个空值,在查blob,好麻烦,用prepareStatment就不用那么麻烦了,哈哈 postblob.heml页...

经验教程

435

收藏

37
微博分享 QQ分享 QQ空间 手机页面 收藏网站 回到头部