探究Android游戏开发之重力系统开发

时间:2011-08-25

  重力我们都很了解的,我们就生活在重力一定的地球上,在外太空重力就不一样了,今天我们讲的就是有关于重力的重力传感器。在重力传感器中,虽然我也实现了一个圆形会根据手机反转的角度而拥有不同的速度,但是其内置加速度算法都是Android os封装好的,而今天我们要讲的重力系统就是去模拟这个加速度,从而让一个自由落体的圆形,感觉跟现实中的皮球一样有质有量!下落的时候速度加快,反弹起来以后速度慢慢减下来。在这里上两张截图,然后简单介绍之后进行讲解。

  当你点击模拟器任意按键的时候会随机在屏幕上生成一个随机大小、随即颜色、随即位置、不停闪烁的一个圆形,并且圆形都拥有重力,在做自由落体,当圆形触到屏幕底部的时候会反弹,并且反弹的高度比低!

  这个实例中,为了好看,我没有让圆形终慢到停下来,会一直在一个高度进行的反弹,下落。

  还有一点:对于圆形当从一个高度自由落体的时候可能它在X坐标系上没有发生改变,当然这是在我们代码中,属于理想状态,因为现实生活中,一般X/Y坐标系都会有变动,在此Demo中,我主要把垂直下落并且反弹的功能做出来了,关于水平的加速度我没做,是因为和垂直的处理思路基本一致。

  MyArc.java

  view plaincopy to clipboardprint?

  package com.himi;

  import java.util.Random;

  import android.graphics.Canvas;

  import android.graphics.Color;

  import android.graphics.Paint;

  import android.graphics.RectF;

  public class MyArc {

  private int arc_x, arc_y, arc_r;

  private float speed_x = 1.2f, speed_y = 1.2f;

  private float vertical_speed;

  private float horizontal_speed;

  private final float ACC = 0.135f;

  private final float RECESSION = 0.2f;

  private boolean isDown = true;

  private Random ran;

  public MyArc(int x, int y, int r) {

  ran = new Random();

  this.arc_x = x;

  this.arc_y = y;

  this.arc_r = r;

  }

  public void drawMyArc(Canvas canvas, Paint paint)

    {

  paint.setColor(getRandomColor());

  canvas.drawArc(new RectF(arc_x + speed_x, arc_y + speed_y, arc_x + 2 *

  arc_r + speed_x, arc_y + 2 * arc_r + speed_y), 0, 360, true, paint);

  }

  public int getRandomColor() {

  int ran_color = ran.nextInt(8);

  int temp_color = 0;

  switch (ran_color) {

  case 0:

  temp_color = Color.WHITE;

  break;

  case 1:

  temp_color = Color.BLUE;

  break;

  case 2:

  temp_color = Color.CYAN;

  break;

  case 3:

  temp_color = Color.DKGRAY;

  break;

  case 4:

  temp_color = Color.RED;

  break;

  case 6:

  temp_color = Color.GREEN;

  case 7:

  temp_color = Color.GRAY;

  case 8:

  temp_color = Color.YELLOW;

  break;

  }

  return temp_color;

  }

  public void logic()

    {

  if (isDown)

    {

  speed_y += vertical_speed;

  int count = (int) vertical_speed++;

  for (int i = 0; i < vertical_speed++; i++) {}

  for (int i = 0; i < count; i++) {

  vertical_speed += ACC;

  }

  } else {

  speed_y -= vertical_speed;

  int count = (int) vertical_speed--;

  for (int i = 0; i < count; i++) {

  vertical_speed -= ACC;

  }

  }

  if (isCollision()) {

  isDown = !isDown;

  vertical_speed -= vertical_speed * RECESSION;

  }

  }

  public boolean isCollision() {

  return arc_y + 2 * arc_r + speed_y >= MySurfaceViee.screenH;

  }

  }

  代码比较简单主要讲解下几个备注:

  备注1:

  估计有些同学看到这里有点小晕,我解释下,大家都知道自由落体的时候,速度是越来越快的,这是受到加速度的影响,所以这里我们对原有的圆形y速度基础上再加上加速度!

  备注2:

  虽然加速度影响了圆形原有的速度,但是我们的加速度也不是恒定的,为了模拟真实球体的自由下落,这里我们不仅对加速度增加了偏移量ACC,而且我们还要对其变化的规律进行模拟,让下次的加速度偏移量成倍增加!所以为什么要for循环的时候把加速度的值当成for循环的一个判定条件!

  好了,下面来看我们SurfaceView!

  view plain <copy">https://blog.csdn.net/xiaominghimi/archive/2011/01/19/6153396.aspx>copy to clipboard <print">https://blog.csdn.net/xiaominghimi/archive/2011/01/19/6153396.aspx>print <https://blog.csdn.net/xiaominghimi/archive/2011/01/19/6153396.aspx><https://blog.csdn.net/xiaominghimi/archive/2011/01/19/6153396.aspx>

  package com.himi;

  import java.util.Random;

  import java.util.Vector;

  import android.content.Context;

  import android.graphics.Canvas;

  import android.graphics.Color;

  import android.graphics.Paint;

  import android.util.Log;

  import android.view.KeyEvent;

  import android.view.SurfaceHolder;

  import android.view.SurfaceView;

  import android.view.SurfaceHolder.Callback;

  public class MySurfaceViee extends SurfaceView implements Callback, Runnable {

  private Thread th;

  private SurfaceHolder sfh;

  private Canvas canvas;

  private Paint paint;

  private boolean flag;

  public static int screenW, screenH;

  private Vector<MyArc> vc;

  private Random ran;

  public MySurfaceViee(Context context) {

  super(context);

  this.setKeepScreenOn(true);

  vc = new Vector<MyArc>();

  ran = new Random();

  sfh = this.getHolder();

  sfh.addCallback(this);

  paint = new Paint();

  paint.setAntiAlias(true);

  setFocusable(true);

  }

  public void surfaceCreated(SurfaceHolder holder) {

  flag = true;

  th = new Thread(this){

  screenW = this.getWidth();

  screenH = this.getHeight();

  th.start();

  }

  public void draw() {

  try {

  canvas = sfh.lockCanvas();

  canvas.drawColor(Color.BLACK);

  if (vc != null) {

  for (int i = 0; i < vc.size(); i++) {

  vc.elementAt(i)。drawMyArc(canvas, paint);

  }

  }

  } catch (Exception e) {

  // TODO: handle exception

  } finally {

  try {

  if (canvas != null)

  sfh.unlockCanvasAndPost(canvas);

  } catch (Exception e2) {

  }

  }

  }

  private void logic() {

  if (vc != null) {

  for (int i = 0; i < vc.size(); i++) {

  vc.elementAt(i)。logic();

  }

  }

  }

  @Override

  public boolean onKeyDown(int keyCode, KeyEvent event) {

  vc.addElement(new MyArc(ran.nextInt(this.getWidth()), ran.nextInt(100), ran.nextInt(50)));

  return true;

  }

  public void run() {

  while (flag) {

  logic();

  draw();

  try {

  Thread.sleep(100);

  } catch (Exception ex) {

  }

  }

  }

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

  Log.v("Himi", "surfaceChanged");

  }

  public void surfaceDestroyed(SurfaceHolder holder) {

  flag = false;

  }

  }

  OK,代码都很简单,也很清晰! 稍微说一句:像MyArc里面也有类似MysurfaceView中一样的方法 logic() 以及draw()这样是更好的管理我们的代码结构,清晰思路,让该干什么的就去干什么,这样省的乱~

  for (int i = 0; i < count; i++) {

  vertical_speed += ACC;

  }

  以上代码确实可以用一句来表示:vertical_speed +=ACC*count; 或者vertical_speed  =vertical_speed + ACC*count;但是要注意:因为我这里变量都是浮点数,大家都知道对于浮点数有位数的限制,那么我这里用for来写可以避免乘积,如果简写的形式会有造成得到的结果有差异!!!!所以要注意;还有千万不要简写成 vertical_speed =(vertical_speed +ACC)*count; 这是错误的!


  
上一篇:一种基于声发射信号在导波杆中传播的放大规律实验研究
下一篇:浅谈电容器测试的挑战与方案

免责声明: 凡注明来源本网的所有作品,均为本网合法拥有版权或有权使用的作品,欢迎转载,注明出处。非本网作品均来自互联网,转载目的在于传递更多信息,并不代表本网赞同其观点和对其真实性负责。

相关技术资料