Android游戏开发之碰撞检测(矩形碰撞、圆形碰撞、像素碰撞)_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;
  }
}

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

以上是小编为您精心准备的的内容,在的博客、问答、公众号、人物、课程等栏目也有的相关内容,欢迎继续使用右上角搜索按钮进行搜索Android碰撞检测
, Android圆形碰撞
Android像素碰撞
圆形和矩形碰撞、android 矩形碰撞检测、矩形碰撞检测算法、cocos2dx像素碰撞检测、旋转矩形碰撞检测,以便于您获取更多的相关知识。

时间: 2024-09-16 15:38:13

Android游戏开发之碰撞检测(矩形碰撞、圆形碰撞、像素碰撞)_Android的相关文章

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

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

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

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

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

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

【ANDROID游戏开发十七】让玩家自定义手势玩转ANDROID游戏!—ANDROID GESTURE之【输入法手势技术】

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

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

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

【ANDROID游戏开发二十二】(图文详解)游戏中灵活实现动画播放!简述J2ME的游戏类库与ANDROID游戏开发!

本站文章均为 李华明Himi 原创,转载务必在明显处注明:  转载自[黑米GameDev街区] 原文链接: http://www.himigame.com/android-game/361.html 由于写书的缘故,博文更新缓慢,大家体谅,今天针对群内常提出动画实现的问题来进行一个详细讲述: 此章节适合没有做过游戏开发的同学学习! 做过Android软件的童鞋们,在学习游戏开发的时候,思维总是被固定在了Android系统组件上!比如动画实现总想着利用BitmapDrawable.Animatio

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

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

【ANDROID游戏开发十八】解放手指,利用传感器开发游戏!(本文讲解在SURFACEVIEW中用重力传感器控制圆球的各方向移动)

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

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

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