热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

Android游戏的基础:物体运动效果

一、新建工程首先,我们在Eclipse中新建一个名为Movement的工程,并且选择合适的AndroidSDK,如下图:新建工程在这里,我们选用的API是比较低的1.5版本,这样可以
一、

  新建工程

  首先,我们在Eclipse中新建一个名为Movement的工程,并且选择合适的Android SDK,如下图:

   新建工程

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

  

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

  

  二、

  编写Movment.java启动程序

  任何一个Android应用都必须有一个主启动程序来启动,我们这里把这个启动程序

  命名为Movment,代码很简单如下:

  

  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及绘图的相关库文件,如下所示:

  

  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的内部接口,可以实现该接口获得界面改变的信息,代码如下,并且我们声明了一些成员变量:

  

  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方向上的初始坐标,如下:

  

  public MovementView(Context context) {

  super(context);

  getHolder().addCallback(this);

  circleRadius = 10;

  circlePaint = new Paint();

  circlePaint.setColor(Color.BLUE);

  xVel = 2;

  yVel = 2;

  }

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

  

  protected void onDraw(Canvas canvas) {

  canvas.drawColor(Color.WHITE);

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

  }

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

  

  public void updatePhysics() {

  //更新当前的x,y坐标

  xPos += xVel;

  yPos += yVel;

  if (yPos - circleRadius<0 || yPos + circleRadius > border="1" Height) {

  if (yPos - circleRadius <0) {

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

  yPos = circleRadius;

  }else{

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

  yPos = border="1" 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线程,代码如下:

  

  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这个方法,这个方法意思是界面尺寸改变时才调用,在我们这个应用中并没用到,所以我们保留为空的方法实现:

  

  public void surfaceChanged(SurfaceHolder holder, int format, int width, int border="1" Height)

  {

  }

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

  

  public void surfaceDestroyed(SurfaceHolder holder) {

  boolean retry = true;

  updateThread.setRunning(false);

  while (retry) {

  try {

  updateThread.join();

  retry = false;

  } catch (InterruptedException e) {

  }

  }

  }

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

  

  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 border="1" 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 > border="1" Height) {

  if (yPos - circleRadius 0) {

  yPos = circleRadius;

  }else{

  yPos = border="1" 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 border="1" Height) {

  }

  public void surfaceDestroyed(SurfaceHolder holder) {

  boolean retry = true;

  updateThread.setRunning(false);

  while (retry) {

  try {

  updateThread.join();

  retry = false;

  } catch (InterruptedException e) {

  }

  }

  }

  }

  四、编写UpdateThread

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

  

  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:

  

  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方法。

  五、运行程序

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


推荐阅读
  • Java容器中的compareto方法排序原理解析
    本文从源码解析Java容器中的compareto方法的排序原理,讲解了在使用数组存储数据时的限制以及存储效率的问题。同时提到了Redis的五大数据结构和list、set等知识点,回忆了作者大学时代的Java学习经历。文章以作者做的思维导图作为目录,展示了整个讲解过程。 ... [详细]
  • 本文介绍了OC学习笔记中的@property和@synthesize,包括属性的定义和合成的使用方法。通过示例代码详细讲解了@property和@synthesize的作用和用法。 ... [详细]
  • Java太阳系小游戏分析和源码详解
    本文介绍了一个基于Java的太阳系小游戏的分析和源码详解。通过对面向对象的知识的学习和实践,作者实现了太阳系各行星绕太阳转的效果。文章详细介绍了游戏的设计思路和源码结构,包括工具类、常量、图片加载、面板等。通过这个小游戏的制作,读者可以巩固和应用所学的知识,如类的继承、方法的重载与重写、多态和封装等。 ... [详细]
  • 本文介绍了使用Java实现大数乘法的分治算法,包括输入数据的处理、普通大数乘法的结果和Karatsuba大数乘法的结果。通过改变long类型可以适应不同范围的大数乘法计算。 ... [详细]
  • 本文讨论了如何优化解决hdu 1003 java题目的动态规划方法,通过分析加法规则和最大和的性质,提出了一种优化的思路。具体方法是,当从1加到n为负时,即sum(1,n)sum(n,s),可以继续加法计算。同时,还考虑了两种特殊情况:都是负数的情况和有0的情况。最后,通过使用Scanner类来获取输入数据。 ... [详细]
  • 本文介绍了如何在给定的有序字符序列中插入新字符,并保持序列的有序性。通过示例代码演示了插入过程,以及插入后的字符序列。 ... [详细]
  • 安卓select模态框样式改变_微软Office风格的多端(Web、安卓、iOS)组件库——Fabric UI...
    介绍FabricUI是微软开源的一套Office风格的多端组件库,共有三套针对性的组件,分别适用于web、android以及iOS,Fab ... [详细]
  • XML介绍与使用的概述及标签规则
    本文介绍了XML的基本概念和用途,包括XML的可扩展性和标签的自定义特性。同时还详细解释了XML标签的规则,包括标签的尖括号和合法标识符的组成,标签必须成对出现的原则以及特殊标签的使用方法。通过本文的阅读,读者可以对XML的基本知识有一个全面的了解。 ... [详细]
  • Google Play推出全新的应用内评价API,帮助开发者获取更多优质用户反馈。用户每天在Google Play上发表数百万条评论,这有助于开发者了解用户喜好和改进需求。开发者可以选择在适当的时间请求用户撰写评论,以获得全面而有用的反馈。全新应用内评价功能让用户无需返回应用详情页面即可发表评论,提升用户体验。 ... [详细]
  • 自动轮播,反转播放的ViewPagerAdapter的使用方法和效果展示
    本文介绍了如何使用自动轮播、反转播放的ViewPagerAdapter,并展示了其效果。该ViewPagerAdapter支持无限循环、触摸暂停、切换缩放等功能。同时提供了使用GIF.gif的示例和github地址。通过LoopFragmentPagerAdapter类的getActualCount、getActualItem和getActualPagerTitle方法可以实现自定义的循环效果和标题展示。 ... [详细]
  • 本文详细介绍了Java中vector的使用方法和相关知识,包括vector类的功能、构造方法和使用注意事项。通过使用vector类,可以方便地实现动态数组的功能,并且可以随意插入不同类型的对象,进行查找、插入和删除操作。这篇文章对于需要频繁进行查找、插入和删除操作的情况下,使用vector类是一个很好的选择。 ... [详细]
  • 猜字母游戏
    猜字母游戏猜字母游戏——设计数据结构猜字母游戏——设计程序结构猜字母游戏——实现字母生成方法猜字母游戏——实现字母检测方法猜字母游戏——实现主方法1猜字母游戏——设计数据结构1.1 ... [详细]
  • [大整数乘法] java代码实现
    本文介绍了使用java代码实现大整数乘法的过程,同时也涉及到大整数加法和大整数减法的计算方法。通过分治算法来提高计算效率,并对算法的时间复杂度进行了研究。详细代码实现请参考文章链接。 ... [详细]
  • Android源码深入理解JNI技术的概述和应用
    本文介绍了Android源码中的JNI技术,包括概述和应用。JNI是Java Native Interface的缩写,是一种技术,可以实现Java程序调用Native语言写的函数,以及Native程序调用Java层的函数。在Android平台上,JNI充当了连接Java世界和Native世界的桥梁。本文通过分析Android源码中的相关文件和位置,深入探讨了JNI技术在Android开发中的重要性和应用场景。 ... [详细]
  • 本文介绍了Web学习历程记录中关于Tomcat的基本概念和配置。首先解释了Web静态Web资源和动态Web资源的概念,以及C/S架构和B/S架构的区别。然后介绍了常见的Web服务器,包括Weblogic、WebSphere和Tomcat。接着详细讲解了Tomcat的虚拟主机、web应用和虚拟路径映射的概念和配置过程。最后简要介绍了http协议的作用。本文内容详实,适合初学者了解Tomcat的基础知识。 ... [详细]
author-avatar
爱看好电影110_275
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有