Android游戏开发之碰撞检测(矩形碰撞、圆形碰撞、像素碰撞)

本文为大家分享了Android游戏开发之碰撞检测,供大家参考,具体内容如下

矩形碰撞 原理: 两个矩形位置 的四种情况 不是这四中情况 则碰撞

圆形碰撞 原理: 利用两个圆心之间的距离进行判定.当两个圆心的距离小于半径之和则碰撞.

像素碰撞 原理:不适用 遍历所有像素 检测 太多了

多矩形碰撞 原理:设置多个矩形碰撞检测区域 检测碰撞矩形数组 与另一碰撞矩形数组之间的位置关系.

矩形碰撞 代码:

public class MySurfaceView extends SurfaceView implements Callback, Runnable {
  private SurfaceHolder sfh;
  private Paint paint;
  private Thread th;
  private boolean flag;
  private Canvas canvas;
  private int screenW, screenH;
  //定义两个矩形的宽高坐标
  private int x1 = 10, y1 = 110, w1 = 40, h1 = 40;
  private int x2 = 100, y2 = 110, w2 = 40, h2 = 40;
  //便于观察是否发生了碰撞设置一个标识位
  private boolean isCollsion;

  /**
   * SurfaceView初始化函数
   */
  public MySurfaceView(Context context) {
    super(context);
    sfh = this.getHolder();
    sfh.addCallback(this);
    paint = new Paint();
    paint.setColor(Color.WHITE);
    paint.setAntiAlias(true);
    setFocusable(true);
  }

  /**
   * SurfaceView视图创建,响应此函数
   */
  @Override
  public void surfaceCreated(SurfaceHolder holder) {
    screenW = this.getWidth();
    screenH = this.getHeight();
    flag = true;
    //实例线程
    th = new Thread(this);
    //启动线程
    th.start();
  }

  /**
   * 游戏绘图
   */
  public void myDraw() {
    try {
      canvas = sfh.lockCanvas();
      if (canvas != null) {
        canvas.drawColor(Color.BLACK);
        if (isCollsion) {
          paint.setColor(Color.RED);
          paint.setTextSize(20);
          canvas.drawText("Collision!", 0, 30, paint);
        } else {
          paint.setColor(Color.WHITE);
        }
        //绘制两个矩形
        canvas.drawRect(x1, y1, x1 + w1, y1 + h1, paint);
        canvas.drawRect(x2, y2, x2 + w2, y2 + h2, paint);
      }
    } catch (Exception e) {
      // TODO: handle exception
    } finally {
      if (canvas != null)
        sfh.unlockCanvasAndPost(canvas);
    }
  }

  /**
   * 触屏事件监听
   */
  @Override
  public boolean onTouchEvent(MotionEvent event) {
    //让矩形1随着触屏位置移动
    x1 = (int) event.getX() - w1 / 2;
    y1 = (int) event.getY() - h1 / 2;
    if (isCollsionWithRect(x1, y1, w1, h1, x2, y2, w2, h2)) {
      isCollsion = true;
    } else {
      isCollsion = false;
    }
    return true;
  }

  /**
   * 按键事件监听
   */
  @Override
  public boolean onKeyDown(int keyCode, KeyEvent event) {
    return super.onKeyDown(keyCode, event);
  }

  /**
   * 游戏逻辑
   */
  private void logic() {

  }

  public boolean isCollsionWithRect(int x1, int y1, int w1, int h1, int x2, int y2, int w2, int h2) {
    if (x1 >= x2 && x1 >= x2 + w2) {
      return false;
    } else if (x1 <= x2 && x1 + w1 <= x2) {
      return false;
    } else if (y1 >= y2 && y1 >= y2 + h2) {
      return false;
    } else if (y1 <= y2 && y1 + h1 <= y2) {
      return false;
    }
    return true;
  }

  @Override
  public void run() {
    while (flag) {
      long start = System.currentTimeMillis();
      myDraw();
      logic();
      long end = System.currentTimeMillis();
      try {
        if (end - start < 50) {
          Thread.sleep(50 - (end - start));
        }
      } catch (InterruptedException e) {
        e.printStackTrace();
      }
    }
  }

  /**
   * SurfaceView视图状态发生改变,响应此函数
   */
  @Override
  public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) {
  }

  /**
   * SurfaceView视图消亡时,响应此函数
   */
  @Override
  public void surfaceDestroyed(SurfaceHolder holder) {
    flag = false;
  }
}

圆形碰撞 代码:

public class MySurfaceView extends SurfaceView implements Callback, Runnable {
  private SurfaceHolder sfh;
  private Paint paint;
  private Thread th;
  private boolean flag;
  private Canvas canvas;
  private int screenW, screenH;
  //定义两个圆形的半径与坐标
  private int r1 = 20, r2 = 20;
  private int x1 = 50, y1 = 100, x2 = 150, y2 = 100;
  //定义一个碰撞标识位
  private boolean isCollision;

  /**
   * SurfaceView初始化函数
   */
  public MySurfaceView(Context context) {
    super(context);
    sfh = this.getHolder();
    sfh.addCallback(this);
    paint = new Paint();
    paint.setColor(Color.WHITE);
    paint.setAntiAlias(true);
    setFocusable(true);
  }

  /**
   * SurfaceView视图创建,响应此函数
   */
  @Override
  public void surfaceCreated(SurfaceHolder holder) {
    screenW = this.getWidth();
    screenH = this.getHeight();
    flag = true;
    //实例线程
    th = new Thread(this);
    //启动线程
    th.start();
  }

  /**
   * 游戏绘图
   */
  public void myDraw() {
    try {
      canvas = sfh.lockCanvas();
      if (canvas != null) {
        canvas.drawColor(Color.BLACK);
        if (isCollision) {
          paint.setColor(Color.RED);
          paint.setTextSize(20);
          canvas.drawText("Collision!", 0, 30, paint);
        } else {
          paint.setColor(Color.WHITE);
        }
        canvas.drawCircle(x1, y1, r1, paint);
        canvas.drawCircle(x2, y2, r2, paint);
      }
    } catch (Exception e) {
      // TODO: handle exception
    } finally {
      if (canvas != null)
        sfh.unlockCanvasAndPost(canvas);
    }
  }

  /**
   * 触屏事件监听
   */
  @Override
  public boolean onTouchEvent(MotionEvent event) {
    x1 = (int) event.getX();
    y1 = (int) event.getY();
    if (isCollisionWithCircle(x1, y1, x2, y2, r1, r2)) {
      isCollision = true;
    } else {
      isCollision = false;
    }
    return true;
  }
  /**
   * 圆形碰撞
   * @param x1  圆形1的圆心X坐标
   * @param y1  圆形2的圆心X坐标
   * @param x2  圆形1的圆心Y坐标
   * @param y2  圆形2的圆心Y坐标
   * @param r1  圆形1的半径
   * @param r2  圆形2的半径
   * @return
   */
  private boolean isCollisionWithCircle(int x1, int y1, int x2, int y2, int r1, int r2) {
    //Math.sqrt:开平方
    //Math.pow(double x, double y): X的Y次方
    if (Math.sqrt(Math.pow(x1 - x2, 2) + Math.pow(y1 - y2, 2)) <= r1 + r2) {
      //如果两圆的圆心距小于或等于两圆半径则认为发生碰撞
      return true;
    }
    return false;
  }

  /**
   * 按键事件监听
   */
  @Override
  public boolean onKeyDown(int keyCode, KeyEvent event) {
    return super.onKeyDown(keyCode, event);
  }

  /**
   * 游戏逻辑
   */
  private void logic() {
  }

  @Override
  public void run() {
    while (flag) {
      long start = System.currentTimeMillis();
      myDraw();
      logic();
      long end = System.currentTimeMillis();
      try {
        if (end - start < 50) {
          Thread.sleep(50 - (end - start));
        }
      } catch (InterruptedException e) {
        e.printStackTrace();
      }
    }
  }

  /**
   * SurfaceView视图状态发生改变,响应此函数
   */
  @Override
  public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) {
  }

  /**
   * SurfaceView视图消亡时,响应此函数
   */
  @Override
  public void surfaceDestroyed(SurfaceHolder holder) {
    flag = false;
  }
}

多矩形碰撞 代码

public class MySurfaceView extends SurfaceView implements Callback, Runnable {
  private SurfaceHolder sfh;
  private Paint paint;
  private Thread th;
  private boolean flag;
  private Canvas canvas;
  private int screenW, screenH;
  //定义两个矩形图形的宽高坐标
  private int rectX1 = 10, rectY1 = 10, rectW1 = 40, rectH1 = 40;
  private int rectX2 = 100, rectY2 = 110, rectW2 = 40, rectH2 = 40;
  //便于观察是否发生了碰撞设置一个标识位
  private boolean isCollsion;
  //定义第一个矩形的矩形碰撞数组
  private Rect clipRect1 = new Rect(0, 0, 15, 15);
  private Rect clipRect2 = new Rect(rectW1 - 15, rectH1 - 15, rectW1, rectH1);
  private Rect[] arrayRect1 = new Rect[] { clipRect1, clipRect2 };
  //定义第二个矩形的矩形碰撞数组
  private Rect clipRect3 = new Rect(0, 0, 15, 15);
  private Rect clipRect4 = new Rect(rectW2 - 15, rectH2 - 15, rectW2, rectH2);
  private Rect[] arrayRect2 = new Rect[] { clipRect3, clipRect4 };

  /**
   * SurfaceView初始化函数
   */
  public MySurfaceView(Context context) {
    super(context);
    sfh = this.getHolder();
    sfh.addCallback(this);
    paint = new Paint();
    paint.setColor(Color.WHITE);
    paint.setAntiAlias(true);
    setFocusable(true);
  }

  /**
   * SurfaceView视图创建,响应此函数
   */
  @Override
  public void surfaceCreated(SurfaceHolder holder) {
    screenW = this.getWidth();
    screenH = this.getHeight();
    flag = true;
    //实例线程
    th = new Thread(this);
    //启动线程
    th.start();
  }

  /**
   * 游戏绘图
   */
  public void myDraw() {
    try {
      canvas = sfh.lockCanvas();
      if (canvas != null) {
        canvas.drawColor(Color.BLACK);
        paint.setColor(Color.WHITE);
        paint.setStyle(Style.FILL);
        if (isCollsion) {
          paint.setTextSize(20);
          canvas.drawText("Collision!", 0, 30, paint);
        }
        //绘制两个矩形
        canvas.drawRect(rectX1, rectY1, rectX1 + rectW1, rectY1 + rectH1, paint);
        canvas.drawRect(rectX2, rectY2, rectX2 + rectW2, rectY2 + rectH2, paint);
        //---绘制碰撞区域使用非填充,并设置画笔颜色白色
        paint.setStyle(Style.STROKE);
        paint.setColor(Color.RED);
        //绘制第一个矩形的所有矩形碰撞区域
        for (int i = 0; i < arrayRect1.length; i++) {
          canvas.drawRect(arrayRect1[i].left + this.rectX1, arrayRect1[i].top + this.rectY1, arrayRect1[i].right + this.rectX1, arrayRect1[i].bottom
              + this.rectY1, paint);
        }
        //绘制第二个矩形的所有矩形碰撞区域
        for (int i = 0; i < arrayRect2.length; i++) {
          canvas.drawRect(arrayRect2[i].left + this.rectX2, arrayRect2[i].top + this.rectY2, arrayRect2[i].right + this.rectX2, arrayRect2[i].bottom
              + rectY2, paint);
        }
      }
    } catch (Exception e) {
      // TODO: handle exception
    } finally {
      if (canvas != null)
        sfh.unlockCanvasAndPost(canvas);
    }
  }

  /**
   * 触屏事件监听
   */
  @Override
  public boolean onTouchEvent(MotionEvent event) {
    //让矩形1随着触屏位置移动
    rectX1 = (int) event.getX() - rectW1 / 2;
    rectY1 = (int) event.getY() - rectH1 / 2;
    if (isCollsionWithRect(arrayRect1, arrayRect2)) {
      isCollsion = true;
    } else {
      isCollsion = false;
    }
    return true;
  }

  /**
   * 按键事件监听
   */
  @Override
  public boolean onKeyDown(int keyCode, KeyEvent event) {
    return super.onKeyDown(keyCode, event);
  }

  /**
   * 游戏逻辑
   */
  private void logic() {

  }

  //Rect 类中的四个属性 top bottom left right
  //分别表示这个矩形的     上    下      左     右
  public boolean isCollsionWithRect(Rect[] rectArray, Rect[] rect2Array) {
    Rect rect = null;
    Rect rect2 = null;
    for (int i = 0; i < rectArray.length; i++) {
      //依次取出第一个矩形数组的每个矩形实例
      rect = rectArray[i];
      //获取到第一个矩形数组中每个矩形元素的属性值
      int x1 = rect.left + this.rectX1;
      int y1 = rect.top + this.rectY1;
      int w1 = rect.right - rect.left;
      int h1 = rect.bottom - rect.top;
      for (int j = 0; j < rect2Array.length; j++) {
        //依次取出第二个矩形数组的每个矩形实例
        rect2 = rect2Array[j];
        //获取到第二个矩形数组中每个矩形元素的属性值
        int x2 = rect2.left + this.rectX2;
        int y2 = rect2.top + this.rectY2;
        int w2 = rect2.right - rect2.left;
        int h2 = rect2.bottom - rect2.top;
        //进行循环遍历两个矩形碰撞数组所有元素之间的位置关系
        if (x1 >= x2 && x1 >= x2 + w2) {
        } else if (x1 <= x2 && x1 + w1 <= x2) {
        } else if (y1 >= y2 && y1 >= y2 + h2) {
        } else if (y1 <= y2 && y1 + h1 <= y2) {
        } else {
          //只要有一个碰撞矩形数组与另一碰撞矩形数组发生碰撞则认为碰撞
          return true;
        }
      }
    }
    return false;
  }

  @Override
  public void run() {
    while (flag) {
      long start = System.currentTimeMillis();
      myDraw();
      logic();
      long end = System.currentTimeMillis();
      try {
        if (end - start < 50) {
          Thread.sleep(50 - (end - start));
        }
      } catch (InterruptedException e) {
        e.printStackTrace();
      }
    }
  }

  /**
   * SurfaceView视图状态发生改变,响应此函数
   */
  @Override
  public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) {
  }

  /**
   * SurfaceView视图消亡时,响应此函数
   */
  @Override
  public void surfaceDestroyed(SurfaceHolder holder) {
    flag = false;
  }
}

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • Android实现消水果游戏代码分享

    消水果游戏大家都玩过吧,今天小编给大家分享实现消水果游戏的代码,废话不多说了,具体代码如下所示: #include "InGameScene.h" #include "PauseLayer.h" #include "ScoreScene.h" #include "AppDelegate.h" extern "C" { void showAds() { } void hideAds() { } } using

  • Android开发之经典游戏贪吃蛇

    前言 这款游戏实现的思路和源码参考了Google自带的Snake的例子,其中修改了一些个人认为还不够完善的地方,加入了一些新的功能,比如屏幕上的方向操作盘,暂停按钮,开始按钮,退出按钮.另外,为了稍微增加些用户体验,除了游戏的主界面,本人自己新增了5个界面,分别是登陆界面,菜单界面,背景音乐设置界面,难度设置界面,还有个关于游戏的介绍界面.个人觉得在新手阶段,参考现成的思路和实现方式是难以避免的.重要的是我们需要有自己的理解,读懂代码之后,需要思考代码背后的实现逻辑,形成自己的思维.这样在下次开

  • Android 重力传感器在游戏开发中的应用

    手势操作可以说是智能手机的一种魅力所在,前两节给大家讲解了两种有趣的手势操作,将它们置于游戏当中,大大提升了游戏的可玩性和趣味性.本节将继续介绍智能手机的另一种神奇之处:传感器.    一.何为传感器 所谓传感器就是能够探测如光.热.温度.重力.方向等等的装置.    二.Android提供了哪些传感器 1.加速度传感器(重力传感器) 2.陀螺仪传感器 3.光传感器 4.恒定磁场传感器 5.方向传感器 6.恒定的压力传感器 7.接近传感器 8.温度传感器 今天我们给大家介绍的是游戏开发中最最常见

  • Android实战打飞机游戏之子弹生成与碰撞以及爆炸效果(5)

    Android实战打飞机游戏子弹生成,新建子弹类 public class Bullet { // 子弹图片资源 public Bitmap bmpBullet; // 子弹的坐标 public int bulletX, bulletY; // 子弹的速度 public int speed; // 子弹的种类以及常量 public int bulletType; // 主角的 public static final int BULLET_PLAYER = -1; // 鸭子的 public st

  • Android实战打飞机游戏之怪物(敌机)类的实现(4)

    先看看效果图: 分析: 根据敌机类型区分 敌机 运动逻辑 以及绘制 /** * 敌机 * * @author liuml * @time 2016-5-31 下午4:14:59 */ public class Enemy { // 敌机的种类标识 public int type; // 苍蝇 public static final int TYPE_FLY = 1; // 鸭子(从左往右运动) public static final int TYPE_DUCKL = 2; // 鸭子(从右往左运

  • Android游戏开发之碰撞检测(矩形碰撞、圆形碰撞、像素碰撞)

    本文为大家分享了Android游戏开发之碰撞检测,供大家参考,具体内容如下 矩形碰撞 原理: 两个矩形位置 的四种情况 不是这四中情况 则碰撞 圆形碰撞 原理: 利用两个圆心之间的距离进行判定.当两个圆心的距离小于半径之和则碰撞. 像素碰撞 原理:不适用 遍历所有像素 检测 太多了 多矩形碰撞 原理:设置多个矩形碰撞检测区域 检测碰撞矩形数组 与另一碰撞矩形数组之间的位置关系. 矩形碰撞 代码: public class MySurfaceView extends SurfaceView imp

  • Android游戏开发学习之引擎用法实例详解

    本文实例讲述了Android游戏开发学习之引擎用法.分享给大家供大家参考.具体如下: 汽车引擎是汽车的心脏,其决定了汽车的性能和稳定性,是人们在购车时相当关注的.而游戏中的物理引擎就如汽车的引擎一样,占据了非常重要的位置.一款好的物理引擎可以非常真实地模拟现实世界,使得游戏更加逼真,提供更好的娱乐体验. 一.JBox2D简介 JBox2D是开源物理引擎Box2D的Java版本,可以直接用于Android.由于JBox2D的图形渲染使用的是Processing库,因此在Android平台上使用JB

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

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

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

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

  • Android游戏开发学习②焰火绽放效果实现方法

    本文实例讲述了Android游戏开发学习②焰火绽放效果实现方法.分享给大家供大家参考.具体如下: 本节介绍在游戏开发中常用到的数学物理应用--粒子系统.粒子系统与上一节的小球有类似的地方,都是通过数学方法和物理公式模拟客观世界中的物体的运动轨迹.不同的是小球更强调个体运动,而焰火粒子等粒子系统更注重整体感觉. 一.焰火粒子效果 1.粒子对象类Particle类和粒子集合类ParticleSet类 每个粒子都为一个Particle类的对象,程序中产生的所有Particle对象都由一个Particl

  • Android 游戏开发中绘制游戏触摸轨迹的曲线图

    本篇文章主要来讲解怎样绘制游戏触摸轨迹的曲线图. 我们在onTouchEvent方法中,可以获取到触摸屏幕时手指触摸点的x.y坐标,如何用这些点形成一条无规则轨迹并把这条无规则轨迹曲线显示在屏幕上就是本篇文章的主旨内容. Android Path类 Android提供了一个Path类 , 顾名思义这个类可以设置曲线路径轨迹.任何无规则的曲线实际上都是由若干条线段组成,而线段的定义为两点之间最短的一条线.path类就 可以记录这两点之间的轨迹,那么若干个Path 就是我们须要绘制的无规则曲线. 下

  • Android游戏开发学习①弹跳小球实现方法

    本文实例讲述了Android游戏开发学习①弹跳小球实现方法.分享给大家供大家参考.具体如下: 在学习了一点点Android之后,觉得有必要记录下来,于是就开了这个新坑,慢慢来填吧. 1.运动体Movable类 本例主要模拟了一组大小不一的球以一定的水平初速度从高处落下的运动轨迹.其中的小球为一个可移动物体Movable对象,该类中除了包含小球图片对象之外,还包括了如位置坐标.水平速度.垂直速度等一系列用于模拟小球运动的成员变量和一些方法. Movable类: package com.ball;

  • Android游戏开发:实现手势操作切换图片的实例

    对于Android 的手势不光在软件中会经常用到,比如浏览器中的翻页,滚动页面等等;当然其实在我们开发Android游戏的时候加上了Android手势操作更会让游戏增加一个亮点,比如一般的CAG.PUZ等类型的游戏选择关卡.简单背景的移动等,都可以使用手势来操作即可,类似前段时间很火的<愤怒的小鸟>,小鸟这个游戏确实不错,我所看到的唯一的亮点是这款游戏的创意!说实话,现在的游戏没有做不出来的只有想不出来的好创意.回到话题来,那么下面我们来了解下什么是Android 手势!        手势识

  • Android游戏开发实践之人物移动地图的平滑滚动处理

    如图所示为程序效果动画图 地图滚动的原理 在本人之前博客的文章中介绍过人物在屏幕中的移动方式,因为之前拼的游戏地图是完全填充整个手机屏幕的,所以无需处理地图的平滑滚动.这篇文章我着重的向 大家介绍一下控制人物移动后地图滚动的处理方式.举个例子 如上图所示 比如人物向右移动,如果地图贴在屏幕左边边界 将先移动人物在地图的坐标,当人物在屏幕中超过三分之二后 则将地图向人物行走的反方向移动给玩家一种人物还在向右移动的假象,其实这时候人物只是播放向右行走的动画 在屏幕中的坐标不变 ,当地图向人物行走反方

  • Android游戏开发之黑白棋

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

随机推荐