Android 游戏开发入门简单示例_Android

 在Android系统上开发游戏是Android开发学习者所向往的,有成就感也有乐趣,还能取得经济上的报酬。那怎样开发Android游戏呢?下面介绍一个简单的入门实例。

       一、创建新工程  

       首先,我们在Eclipse中新建一个名为Movement的工程,并且选择合适的Android SDK,在这里,我们选用的API是比较低的1.5版本,这样可以让其适应性更强。接下来,我们新建两个类,一个是UpdateThread类,一个是SurfaceView类,它们在项目中分别是负责处理线程和画面的两个类,在接下来会有详细介绍,如下图,分别建立这两个类,注意选择正确它们继承的父类:

       在建立完成后,系统的项目结构看上去应该象如下的样子:

       二、编写Movment.java启动程序

       任何一个Android应用都必须有一个主启动程序来启动,我们这里把这个启动程序命名为Movment,代码很简单如下:

Java代码

public class Movement extends Activity {
 @Override
 public void onCreate(Bundle savedInstanceState) {  

   super.onCreate(savedInstanceState);
   setContentView(new MovementView(this));
 }  

       注意的是,我们这个启动程序不象其他程序一样,在启动的时候,在setContentView中传入界面布局文件,而是直接将MovementView的实例传递进来,也就是说,直接启动了MovementView这个类,在这个类中,我们将绘画我们的小球。

       三、什么是SurfaceView

       在Android中,SurfaceView是一个重要的绘图容器,它可以可以直接从内存或者DMA等硬件接口取得图像数据。通常情况程序的View和用户响应都是在同一个线程中处理的,这也是为什么处理长时间事件(例如访问网络)需要放到另外的线程中去(防止阻塞当前UI线程的操作和绘制)。但是在其他线程中却不能修改UI元素,例如用后台线程更新自定义View(调用View的在自定义View中的onDraw函数)是不允许的。

       如果需要在另外的线程绘制界面、需要迅速的更新界面或则渲染UI界面需要较长的时间,这种情况就要使用SurfaceView了。SurfaceView中包含一个Surface对象,而Surface是可以在后台线程中绘制的。

       在本文中,我们将使用它,直接通过代码创建一个小球,并且随着UpdateThread线程的更新,不断改变小球的位置,下面我们开始学习MovementView的编写,先看下如何运用SurfaceView。

       首先导入SurfaceView及绘图的相关库文件,如下所示:

Java代码

package example.movement;  

import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.graphics.Rect;
import android.view.SurfaceHolder;
import android.view.SurfaceView; 

       接着,我们要继承SurfaceView并且实现SurfaceHolder.Callback接口,这是一个SurfaceHolder的内部接口,可以实现该接口获得界面改变的信息,代码如下,并且我们声明了一些成员变量:

Java代码

public class MovementView extends SurfaceView implements SurfaceHolder.Callback {
  private int xPos;
  private int yPos;  

  private int xVel;
  private int yVel;  

  private int width;
  private int height;  

  private int circleRadius;
  private Paint circlePaint;  

  UpdateThread updateThread;
}

        而在MovementView的构造函数中,我们设置了小球的大小和在X,Y方向上的初始坐标,如下: 

Java代码         

public MovementView(Context context) {
  super(context);
  getHolder().addCallback(this);   

  circleRadius = 10;
  circlePaint = new Paint();
  circlePaint.setColor(Color.BLUE);   

  xVel = 2;
  yVel = 2;
}

       接着我们来看下ondraw方法的编写,在这里,我们将绘画小球,并且每次都把画布Canvas的背景色设置为白色,以重新覆盖之前一帧,代码如下:

Java代码

protected void onDraw(Canvas canvas) {  

    canvas.drawColor(Color.WHITE);  

    canvas.drawCircle(xPos, yPos, circleRadius, circlePaint);
}

       我们再来看下updatePhysics这个方法如何编写。这个方法的作用有两个:一是处理小球的运动,二是更新小球的实时位置,因为小球在屏幕中不断地运动,因此当小球到达比如屏幕绘画区域的顶端后,要被弹回,因此代码如下:

Java代码

public void updatePhysics() {  

//更新当前的x,y坐标
    xPos += xVel;
    yPos += yVel;  

    if (yPos - circleRadius < 0 || yPos + circleRadius > height) {  

      if (yPos - circleRadius < 0) {  

        //如果小球到达画布区域的上顶端,则弹回  

        yPos = circleRadius;
      }else{  

        //如果小球到达了画布的下端边界,则弹回  

        yPos = height - circleRadius;
      }  

      // 将Y坐标设置为相反方向
      yVel *= -1;
    }
    if (xPos - circleRadius < 0 || xPos + circleRadius > width) {  

      if (xPos - circleRadius < 0) {  

        // 如果小球到达左边缘  

        xPos = circleRadius;
      } else {  

        // 如果小球到达右边缘  

        xPos = width - circleRadius;
      }  

      // 重新设置x轴坐标
      xVel *= -1;
    }
  }

       最后我们看下surfaceCreated这个方法的代码,在这个方法中,主要是取得了可用的SurfaceView的区域的高度和宽度,然后设置了小球的起始坐标(将其设置在屏幕的正中央位置),并且启动了UpdateThread线程,代码如下:

Java代码

public void surfaceCreated(SurfaceHolder holder) {  

    Rect surfaceFrame = holder.getSurfaceFrame();
    width = surfaceFrame.width();
    height = surfaceFrame.height();  

    xPos = width / 2;
    yPos = circleRadius;  

    updateThread = new UpdateThread(this);
    updateThread.setRunning(true);
    updateThread.start();
  }

       此外,我们要补上surfaceChanged这个方法,这个方法意思是界面尺寸改变时才调用,在我们这个应用中并没用到,所以我们保留为空的方法实现:

Java代码

public void surfaceChanged(SurfaceHolder holder, int format, int width, int height)
  {  

  }  

       而surfaceDestroyed方法中,主要实现的是界面被销毁时才调用,这里我们停止了当前的线程所处理的任务,这里使用了线程的join方法:

Java代码

public void surfaceDestroyed(SurfaceHolder holder) {  

    boolean retry = true;  

    updateThread.setRunning(false);
    while (retry) {
      try {
        updateThread.join();
        retry = false;
      } catch (InterruptedException e) {  

      }
    }
  }

       归纳下,完整的MovementView代码如下:

Java代码

package example.movement;  

import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.graphics.Rect;
import android.view.SurfaceHolder;
import android.view.SurfaceView;  

public class MovementView extends SurfaceView implements SurfaceHolder.Callback {  

  private int xPos;
  private int yPos;  

  private int xVel;
  private int yVel;  

  private int width;
  private int height;  

  private int circleRadius;
  private Paint circlePaint;  

  UpdateThread updateThread;  

  public MovementView(Context context) {  

    super(context);
    getHolder().addCallback(this);  

    circleRadius = 10;
    circlePaint = new Paint();
    circlePaint.setColor(Color.BLUE);  

    xVel = 2;
    yVel = 2;
  }
  @Override
  protected void onDraw(Canvas canvas) {  

    canvas.drawColor(Color.WHITE);
    canvas.drawCircle(xPos, yPos, circleRadius, circlePaint);
  }  

  public void updatePhysics() {
    xPos += xVel;
    yPos += yVel;  

    if (yPos - circleRadius < 0 || yPos + circleRadius > height) {
      if (yPos - circleRadius < 0) {
        yPos = circleRadius;
      }else{
        yPos = height - circleRadius;
      }
      yVel *= -1;
    }
    if (xPos - circleRadius < 0 || xPos + circleRadius > width) {
      if (xPos - circleRadius < 0) {
        xPos = circleRadius;
      } else {
        xPos = width - circleRadius;
      }
      xVel *= -1;
    }
  }  

  public void surfaceCreated(SurfaceHolder holder) {  

    Rect surfaceFrame = holder.getSurfaceFrame();
    width = surfaceFrame.width();
    height = surfaceFrame.height();  

    xPos = width / 2;
    yPos = circleRadius;  

    updateThread = new UpdateThread(this);
    updateThread.setRunning(true);
    updateThread.start();
  }  

  public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) {
  }  

  public void surfaceDestroyed(SurfaceHolder holder) {  

    boolean retry = true;  

    updateThread.setRunning(false);
    while (retry) {
      try {
        updateThread.join();
        retry = false;
      } catch (InterruptedException e) {
      }
    }
  }
}

         四、UpdateThread线程程序

       下面,我们开始着手编写UpdateThread线程程序。这个程序主要是启动一个线程去不断更新当前小球的位置。先看声明及构造函数部分:

Java代码

package licksquid.movement;  

import android.graphics.Canvas;
import android.view.SurfaceHolder;  

public class UpdateThread extends Thread {
  private long time;
  private final int fps = 20;
  private boolean toRun = false;
  private MovementView movementView;
  private SurfaceHolder surfaceHolder;  

}
public UpdateThread(MovementView rMovementView) {
    movementView = rMovementView;
    surfaceHolder = movementView.getHolder();
  }
 public void setRunning(boolean run) {
    toRun = run;
  }

        注意这里的setRunning方法中设置了线程是否应该停止的标记,下面来看重要的方法run:

Java代码

public void run() {  

    Canvas c;
    while (toRun) {  

      long cTime = System.currentTimeMillis();  

      if ((cTime - time) <= (1000 / fps)) {  

        c = null;
        try {
          c = surfaceHolder.lockCanvas(null);  

          movementView.updatePhysics();
          movementView.onDraw(c);
        } finally {
          if (c != null) {
            surfaceHolder.unlockCanvasAndPost(c);
          }
        }
      }
      time = cTime;
    }
  }

       在run方法中,主要实现了如下几个任务:首先检查是否有允许启动该线程(在开始运行后,由于在MovementView中,启动UpdateThread的时候,已经设置了其值为true,即updateThread.setRunning(true)),接下来检查是否在指定的时间内(这里设置的是每秒20帧),如果是的话,则调用surfaceHolder的lockCanvas方法,锁定当前的画布绘画区域,并且调用movementView的updatePhysics方法及onDraw方法去画小球并判断小球的运动,最后记得要在finally中调用unlockCanvasAndPost方法。

       五、启动并运行程序

       最后启动并运行程序,可以看到如下的效果,可以看到小球在做各个方向的弹跳运动。

       到此就完成了这个Android游戏开发的入门实例,其实编写Android游戏就是这么简单。

        以上就是简单的游戏开发程序,后续继续整理相关知识,谢谢大家对本站的支持!

以上是小编为您精心准备的的内容,在的博客、问答、公众号、人物、课程等栏目也有的相关内容,欢迎继续使用右上角搜索按钮进行搜索android
, 游戏开发
, 游戏
游戏开发示例
jmx入门及开发示例、android开发示例、android商城开发示例、android开发入门、android开发入门教程,以便于您获取更多的相关知识。

时间: 2024-09-10 21:10:12

Android 游戏开发入门简单示例_Android的相关文章

Android 游戏开发入门简单示例

在Android系统上开发游戏是Android开发学习者所向往的,有成就感也有乐趣,还能取得经济上的报酬.那怎样开发Android游戏呢?下面介绍一个简单的入门实例. 一.创建新工程 首先,我们在Eclipse中新建一个名为Movement的工程,并且选择合适的Android SDK,在这里,我们选用的API是比较低的1.5版本,这样可以让其适应性更强.接下来,我们新建两个类,一个是UpdateThread类,一个是SurfaceView类,它们在项目中分别是负责处理线程和画面的两个类,在接下来

Android中资源文件用法简单示例_Android

本文实例讲述了Android中资源文件用法.分享给大家供大家参考,具体如下: 一.XML文件间资源文件的使用 引用格式: attribute="@[packagename:]resourcetype/resourceidentifier" example: 有strings.xml.color.xml.dimen.xml文件,使用其中的参数 <?xml version="1.0" encoding="utf-8"?> <Line

Android游戏开发之黑白棋_Android

黑白棋介绍 黑白棋,又叫苹果棋,最早流行于西方国家.游戏通过相互翻转对方的棋子,最后以棋盘上谁的棋子多来判断胜负.黑白棋非常易于上手,但精通则需要考虑许多因素,比如角边这样的特殊位置.稳定度.行动力等.本游戏取名为黑白棋大师,提供了8种难度等级的选择,从菜鸟.新手.入门.棋手到棋士.大师.宗师.棋圣,助你不断提升棋力. 黑白棋游戏规则 游戏规则见黑白棋大师中的截图. 黑白棋大师游戏截图 游戏启动界面. 游戏过程中的一个截图. 开新局时的选项,选择先后手以及AI的水平. 几个关键的类 Rule R

Android游戏开发 自定义手势--输入法手势技术_Android

       进行软件开发时,通常我们都喜欢使用较新版本的工具,但这里我为什么使用低版本的SDK来开发Android游戏呢?这里介绍下原因:        1.Android SDK 属于向下兼容!那么低版本可以运行的,高版本基本上更是没问题!(当然每次SDK的更新也会带来新功能,或者修改了一些原来的BUG等等,那么其实对于游戏开发来说,如果你的游戏中不需要更高的SDK版本的支持情况下,完全不必去追求最新的SDK!)        2.使用低版本进行游戏开发这样能兼顾更多的机型,获取更多的用户!

《Android游戏开发详解》一2.7 构建一个简单的计数程序

2.7 构建一个简单的计数程序 Android游戏开发详解在下一个示例中,我们将利用第1章中介绍过的for循环来打印出数字5到12之间的每一个偶数.这是一个简单的游戏示例,但是,掌握for循环语法的技巧很重要. 创建一个名为CountingProject的新的Java项目,并且创建一个名为EvenFinder的新类,添加程序清单2.7所示的main方法. 程序清单2.7 EvenFinder类 01 public class EvenFinder { 02 03 public static vo

【ANDROID游戏开发十六】ANDROID GESTURE之【触摸屏手势识别】操作!利用触摸屏手势实现一个简单切换图片的功能!

本站文章均为 李华明Himi 原创,转载务必在明显处注明:  转载自[黑米GameDev街区] 原文链接: http://www.himigame.com/android-game/337.html ----------------------- 『很多童鞋说我的代码运行后,点击home或者back后会程序异常,如果你也这样遇到过,那么你肯定没有仔细读完Himi的博文,第十九篇Himi专门写了关于这些错误的原因和解决方法,这里我在博客都补充说明下,省的童鞋们总疑惑这一块:请点击下面联系进入阅读:

【ANDROID游戏开发之一】设置全屏以及绘画简单的图形

本站文章均为 李华明Himi 原创,转载务必在明显处注明:  转载自[黑米GameDev街区] 原文链接: http://www.himigame.com/android-game/263.html 『很多童鞋说我的代码运行后,点击home或者back后会程序异常,如果你也这样遇到过,那么你肯定没有仔细读完Himi的博文,第十九篇Himi专门写了关于这些错误的原因和解决方法,这里我在博客都补充说明下,省的童鞋们总疑惑这一块:请点击下面联系进入阅读: [Android游戏开发十九](必看篇)Sur

【ANDROID游戏开发二十一】ANDROID OS设备谎言分辨率的解决方案!以及简单阐述游戏引擎如何使用!

本站文章均为 李华明Himi 原创,转载务必在明显处注明:  转载自[黑米GameDev街区] 原文链接: http://www.himigame.com/android-game/356.html 刚才一群里的兄弟问的一问题,稍微研究下,这里一起分享:新建的Emulator -配置为:WAGA800  其分辨率是 800*480 的设备模拟器,当我们程序中在取得其 Height和 Width的时候发现,总是 320*533 ,明显是系统对我们撒了谎!如下图:   下面是官方文档原文: http

《Android游戏开发详解》一1.4 关于位的一切(位和字节的简单介绍)

1.4 关于位的一切(位和字节的简单介绍) Android游戏开发详解 在我们继续深入之前,值得先细致地介绍如何具体把值存储到变量中.我前面提到,不同的基本数据类型具有不同的位大小.例如,一个int有32位而一个long有64位.你可能会问,那么,到底什么是位? 位(bit)是一个二进制位的简称.换句话说,如果你有一个只有0和1的二进制数,每个数字就是1位.达到8位的时候,例如,(10101001),你就有了1字节. 对于位,你需要记住的一点是:拥有的位越多,所能表示的数值也越多.为了说明这一点