【读书笔记《Android游戏编程之从零开始》】18.游戏开发基础(碰撞检测)

简介:
1.矩形碰撞
所谓矩形碰撞,就是利用两个矩形之间的位置关系来进行判断,如果矩形的像素在另外一个矩形之中,或者之上都可以认为这两个矩形发生了碰撞。
如果单纯的去考虑哪些情况会判定两个矩形发生碰撞,倒不如反思考虑两个矩形之间不发生碰撞的几种情况。其实两个矩形不发生碰撞的情况就上下左右这四种。
下面通过实例项目来完成对应的四种判定。
先看实例效果图:
新建项目,游戏框架为 SurfaceView 游戏框架,修改 MySurfaceView 类如下:
  View Code
 
PS:在两个矩形之间进行碰撞检测时,不仅仅要判定两者X、Y坐标之间的位置关系,还要考虑到两个矩形的宽度和高度。
 
2.圆形碰撞
圆形之间的碰撞,主要是利用两圆心的圆心距离进行判定的;当两圆心的圆心距小于两圆半径之和时,判定发生了碰撞。还是用一个实例进行说明,效果图如下:
 
 
 
新建项目,游戏框架为 SurfaceView 游戏框架,修改 MySurfaceView 类如下,其中主要就是圆形碰撞函数方法isCollisionWithCircle:
 
  View Code

 

3.像素碰撞
对于碰撞检测已经介绍了矩形与圆形两种方式,其实使用这两种检测方式不是很精确。比如两张大小相同的带透明度的矩形png图,如果利用矩形来进行碰撞,肯定不能以图的大小进行碰撞检测,因为可能只是透明像素位置发生碰撞,而非透明像素并没有发生碰撞,在屏幕上直观看来,也并没有发生碰撞。
而像素碰撞是怎样模拟碰撞的呢?首先遍历算出一张位图所有的像素点坐标,然后与另外一张位图上的所有点坐标进行对比,一旦有一个像素点的坐标相同,就立刻取出这两个坐标相同的像素点,通过位运算取出这两个像素点的最高位(透明度)进行对比,如果两个像素点都是非透明像素则判定这两张位图发生碰撞。
 
但正因为像素碰撞的这种高精确性判定,从而也会造成代码效率明显降低!假设两张100×100 大小的位图利用像素级检测碰撞,仅是遍历两张位图的像素点就要循环 100×100×2 = 20000 句逻辑代码;况且还要筛选出来的相同坐标的像素点进行遍历对比其透明值!这中效率可想而知!
像素级的碰撞检测是不推荐使用的。一般游戏开发中,取代像素级碰撞检测的方法是利用“多矩形”、“多圆形”的检测方式来实现的。

4.多矩形碰撞
所谓多矩形碰撞,顾名思义就是设置多个矩形碰撞区域。还是用一个实例进行说明,矩形部分为非透明部分,其他为透明部分,效果图如下:
 
 新建项目,游戏框架为 SurfaceView 游戏框架,修改 MySurfaceView 类如下
  View Code

 

由于多圆形的碰撞检测类似于多矩形碰撞,不再赘述。

 

5.Region 碰撞检测

在之前的笔记“ 游戏开发基础(剪切区域)”中介绍过 Region 这个类,其实此类还有一个比较常用的方法就是用于判断一个点是否在矩形区域内,其方法是使用 Region 类中的 contains(int x,int y) 函数。

下面用一个实例进行说明,效果为当用户触屏的坐标点在碰撞矩形内时,显示指定的位图:

 

 新建项目,游戏框架为 SurfaceView 游戏框架,修改 MySurfaceView 类如下

复制代码
import android.content.Context;
import android.graphics.BitmapFactory;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.graphics.Rect;
import android.graphics.Region;
import android.view.KeyEvent;
import android.view.MotionEvent;
import android.view.SurfaceHolder;
import android.view.SurfaceHolder.Callback;
import android.view.SurfaceView;

/**
 * 
 * @author David
 * 
 */
public class MySurfaceView extends SurfaceView implements Callback, Runnable {
    // 用于控制SurfaceView 的大小、格式等,并且主要用于监听SurfaceView 的状态
    private SurfaceHolder sfh;
    // 声明一个画笔
    private Paint paint;
    // 声明一个画布
    private Canvas canvas;
    // 声明一个线程
    private Thread th;
    // 线程消亡的标识符
    private boolean flag;
    // 声明屏幕的宽高
    private int screenW, screenH;

    // 定义碰撞矩形
    private Rect rect = new Rect(0, 0, 50, 50);
    // 定义Region类实例
    private Region r = new Region(rect);
    // 表示是否发生碰撞的标识位
    private boolean isInclude;

    /**
     * SurfaceView 初始化函数
     * 
     * @param context
     */
    public MySurfaceView(Context context) {
        super(context);
        // 实例SurfaceView
        sfh = this.getHolder();
        // 为SurfaceView添加状态监听
        sfh.addCallback(this);
        // 实例一个画笔
        paint = new Paint();
        // 设置字体大小
        paint.setTextSize(20);
        // 设置画笔的颜色
        paint.setColor(Color.WHITE);
        // 设置画布无锯齿
        paint.setAntiAlias(true);
        // 设置焦点
        setFocusable(true);
    }

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

    }

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

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

    /**
     * 游戏绘图
     */
    private void myDraw() {
        try {
            // 获取SurfaceView 的Canvas 对象,
            canvas = sfh.lockCanvas();
            if (canvas != null) {
                // 刷新画布,背景为黑色
                canvas.drawColor(Color.BLACK);
                // 标识位为真时,绘制icon图
                if (isInclude) {
                    canvas.drawBitmap(BitmapFactory.decodeResource(
                            this.getResources(), R.drawable.ic_launcher), 100,
                            50, paint);
                }
                // 绘制矩形区域(便于观察)
                canvas.drawRect(rect, paint);
            }
        } catch (Exception e) {

        } finally {
            if (canvas != null) {
                // 解锁画布和提交
                sfh.unlockCanvasAndPost(canvas);
            }
        }
    }

    /**
     * 触屏事件监听
     */
    @Override
    public boolean onTouchEvent(MotionEvent event) {
        // 判定用户触屏的坐标点是否在碰撞矩形内
        if (r.contains((int) event.getX(), (int) event.getY())) {
            isInclude = true;
        } else {
            isInclude = false;
        }
        return true;
    }

    @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 (Exception e) {
                e.printStackTrace();
            }
        }
    }
}
复制代码





本文转自叶超Luka博客园博客,原文链接:http://www.cnblogs.com/yc-755909659/p/4186193.html,如需转载请自行联系原作者
目录
相关文章
|
1月前
|
设计模式 Java Android开发
安卓应用开发中的内存泄漏检测与修复
【9月更文挑战第30天】在安卓应用开发过程中,内存泄漏是一个常见而又棘手的问题。它不仅会导致应用运行缓慢,还可能引发应用崩溃,严重影响用户体验。本文将深入探讨如何检测和修复内存泄漏,以提升应用性能和稳定性。我们将通过一个具体的代码示例,展示如何使用Android Studio的Memory Profiler工具来定位内存泄漏,并介绍几种常见的内存泄漏场景及其解决方案。无论你是初学者还是有经验的开发者,这篇文章都将为你提供实用的技巧和方法,帮助你打造更优质的安卓应用。
|
6月前
|
Android开发
如何在Android真机上检测是否有Google Map add-on
如何在Android真机上检测是否有Google Map add-on
71 3
|
4月前
|
监控 Java Android开发
探究Android应用开发中的内存泄漏检测与修复
在移动应用的开发过程中,优化用户体验和提升性能是至关重要的。对于Android平台而言,内存泄漏是一个常见且棘手的问题,它可能导致应用运行缓慢甚至崩溃。本文将深入探讨如何有效识别和解决内存泄漏问题,通过具体案例分析,揭示内存泄漏的成因,并提出相应的检测工具和方法。我们还将讨论一些最佳实践,帮助开发者预防内存泄漏,确保应用稳定高效地运行。
|
6月前
|
Android开发
android检测网络连接是否存在(一)
android检测网络连接是否存在(一)
45 2
|
6月前
|
Android开发
安卓手机快速过检测完成某某学习
安卓手机快速过检测完成某某学习
53 0
|
6月前
|
XML 监控 Java
Android App开发之事件交互Event中检测软键盘和物理按键讲解及实战(附源码 演示简单易懂)
Android App开发之事件交互Event中检测软键盘和物理按键讲解及实战(附源码 演示简单易懂)
699 0
|
Android开发 UED
Android版本检测更新升级
Android版本检测更新升级
|
API Android开发
使用Android的Service实现后台定时检测并重启应用
使用Android的Service实现后台定时检测并重启应用
|
Android开发
全网最优雅安卓列表项可见性检测
全网最优雅安卓列表项可见性检测
172 0
|
算法 数据挖掘 Android开发
全网最优雅安卓控件可见性检测
全网最优雅安卓控件可见性检测
227 0