Unity实现图形相交检测
作者:小混沌 发布时间:2021-09-16 09:31:01
前言
图形相交检测常常用在伤害判定,使用自定义的图形相交检测,可以在一定程度上控制性能。
比如2D格斗游戏中使用的矩形包围盒(AABB),一些动作游戏中常常出现的扇形攻击。
2D的图形相交检测能够满足大部分的需求,且可以拓展成为柱状的3D物体,2D比3D的计算复杂度会低很多,3D的图形检测原理与2D相似,本文会实现几个圆形与其他2D图形的相交检测:
1、圆形与圆形
2、圆形与胶囊体
3、圆形与扇形
4、圆形与凸多边形
5、圆形与AABB
6、圆形与OBB
通过简单化处理,把被判定物都处理成由圆柱或多个圆柱构成的区域,所以只需要考虑圆形与其他形状的相交。
圆形与圆形
两个圆形的相交检测非常简单直观,只需要判断半径只和与距离的大小。
定义圆形区间:
/// <summary>
/// 圆形区间
/// </summary>
public struct CircleArea
{
public Vector2 o;
public float r;
}
o ——圆心坐标
r ——圆半径
相交判断:
/// <summary>
/// 判断圆形与圆形相交
/// </summary>
/// <param name="circleArea"></param>
/// <param name="target"></param>
/// <returns></returns>
public static bool Circle(CircleArea circleArea, CircleArea target)
{
return (circleArea.o - target.o).sqrMagnitude < (circleArea.r + target.r) * (circleArea.r + target.r);
}
分离轴定理
分离轴定理(separating axis theorem, SAT)分离轴定理是指,两个不相交的凸集必然存在一个分离轴,使两个凸集在该轴上的投影是分离的。
判断两个形状是否相交,实际上是判断分离轴是否能把两个形状分离。若存在分离轴能使两个图形分离,则这两个图形是分离的。
基于以上理论,寻找分离轴是我们要做的工作,重新考虑两个圆形的相交检测,实际上我们做的是把圆心连线的方向作为分离轴:
上图中两图形的投影在分离轴上是分离的,存在分离线将两者隔开,于是我们可以断定两图形是分离的。
胶囊体的本质
定义一个线段 u,距离 d。胶囊体实际上是与线段 u 的最短距离小于 d 的点的集合。判断一个点 x 处于胶囊体内部,就是判断点与线段的距离。
求点 x 与线段 u 最短距离的过程是:
1、求出点 x 在线段 u 所在直线上的投影点 P;
2、将投影点 P 限制在线段的范围内(如右图中投影点不在线段内,则限定到线段内);
3、x 与 P 的距离即为所求;
/// <summary>
/// 线段与点的最短距离。
/// </summary>
/// <param name="x0">线段起点</param>
/// <param name="u">线段向量</param>
/// <param name="x">求解点</param>
/// <returns></returns>
public static float SqrDistanceBetweenSegmentAndPoint(Vector2 x0, Vector2 u, Vector2 x)
{
float t = Vector2.Dot(x - x0, u) / u.sqrMagnitude;
return (x - (x0 + Mathf.Clamp01(t) * u)).sqrMagnitude;
}
为避免开方计算,结果使用距离的平方。
圆形与胶囊体
分离轴是线段上距离圆心最近的点P与圆心所在方向。
定义胶囊体:
/// <summary>
/// 胶囊体
/// </summary>
public struct CapsuleArea
{
public Vector2 X0;
public Vector2 U;
public float d;
}
相交判断:
/// <summary>
/// 判断胶囊体与圆形相交
/// </summary>
/// <param name="capsuleArea"></param>
/// <param name="circleArea"></param>
/// <returns></returns>
public static bool Capsule(CapsuleArea capsuleArea, CircleArea circleArea)
{
float sqrD = SegmentPointSqrDistance(capsuleArea.X0, capsuleArea.U, circleArea.o);
return sqrD < (circleArea.r + capsuleArea.d) * (circleArea.r + capsuleArea.d);
}
圆形与扇形
当扇形角度大于180度时,就不再是凸多边形了,不能适用于分离轴理论。我们可以找出相交时圆心的所有可能区域,并把区域划分成可以简单验证的几个区域,逐个试验。
这里共划分了2个区间
1、半径为两者半径和的扇形区间,角度方向同扇形。验证方法是;验证距离与夹角。
2、扇形边为轴,圆形半径为大小组成的胶囊体空间,由于扇形的对称性,我们可以通过把圆心映射到一侧,从而只需要计算1条边。
定义扇形:
/// <summary>
/// 扇形区间。
/// </summary>
public struct SectorArea
{
public Vector2 o;
public float r;
public Vector2 direction;
public float angle;
}
相交检测:
/// <summary>
/// 判断圆形与扇形相交。
/// </summary>
/// <param name="sectorArea"></param>
/// <param name="target"></param>
/// <returns></returns>
public static bool Sector(SectorArea sectorArea, CircleArea target)
{
Vector2 tempDistance = target.o - sectorArea.o;
float halfAngle = Mathf.Deg2Rad * sectorArea.angle / 2;
if (tempDistance.sqrMagnitude < (sectorArea.r + target.r) * (sectorArea.r + target.r))
{
if (Vector3.Angle(tempDistance, sectorArea.direction) < sectorArea.angle / 2)
{
return true;
}
else
{
Vector2 targetInSectorAxis = new Vector2(Vector2.Dot(tempDistance,
sectorArea.direction), Mathf.Abs(Vector2.Dot(tempDistance, new Vector2(-sectorArea.direction.y, sectorArea.direction.x))));
Vector2 directionInSectorAxis = sectorArea.r * new Vector2(Mathf.Cos(halfAngle), Mathf.Sin(halfAngle));
return SegmentPointSqrDistance(Vector2.zero, directionInSectorAxis, targetInSectorAxis) <= target.r * target.r;
}
}
return false;
}
圆形与凸多边形
定义多边形:
/// <summary>
/// 多边形区域。
/// </summary>
public struct PolygonArea
{
public Vector2[] vertexes;
}
相交检测:
/// <summary>
/// 判断多边形与圆形相交
/// </summary>
/// <param name="polygonArea"></param>
/// <param name="target"></param>
/// <returns></returns>
public static bool PolygonS(PolygonArea polygonArea, CircleArea target)
{
if (polygonArea.vertexes.Length < 3)
{
Debug.Log("多边形边数小于3.");
return false;
}
#region 定义临时变量
//圆心
Vector2 circleCenter = target.o;
//半径的平方
float sqrR = target.r * target.r;
//多边形顶点
Vector2[] polygonVertexes = polygonArea.vertexes;
//圆心指向顶点的向量数组
Vector2[] directionBetweenCenterAndVertexes = new Vector2[polygonArea.vertexes.Length];
//多边形的边
Vector2[] polygonEdges = new Vector2[polygonArea.vertexes.Length];
for (int i = 0; i < polygonArea.vertexes.Length; i++)
{
directionBetweenCenterAndVertexes[i] = polygonVertexes[i] - circleCenter;
polygonEdges[i] = polygonVertexes[i] - polygonVertexes[(i + 1)% polygonArea.vertexes.Length];
}
#endregion
#region 以下为圆心处于多边形内的判断。
//总夹角
float totalAngle = Vector2.SignedAngle(directionBetweenCenterAndVertexes[polygonVertexes.Length - 1], directionBetweenCenterAndVertexes[0]);
for (int i = 0; i < polygonVertexes.Length - 1; i++)
totalAngle += Vector2.SignedAngle(directionBetweenCenterAndVertexes[i], directionBetweenCenterAndVertexes[i + 1]);
if (Mathf.Abs(Mathf.Abs(totalAngle) - 360f) < 0.1f)
return true;
#endregion
#region 以下为多边形的边与圆形相交的判断。
for (int i = 0; i < polygonEdges.Length; i++)
if (SegmentPointSqrDistance(polygonVertexes[i], polygonEdges[i], circleCenter) < sqrR)
return true;
#endregion
return false;
}
圆形与AABB
定义AABB:
/// <summary>
/// AABB区域
/// </summary>
public struct AABBArea
{
public Vector2 center;
public Vector2 extents;
}
AABB是凸多边形的特例,是长宽边分别与X/Y轴平行的矩形,这里我们要充分的利用他的对称性。
1 利用对称性将目标圆心映射到,以AABB中心为原点、两边为坐标轴的坐标系,的第一象限
2 将目标圆心映射到,以AABB第一象限角点为原点、两边为坐标轴的坐标系,的第一象限
3 最后只需要判断圆形半径与步骤2中映射点的向量大小
相交检测:
/// <summary>
/// 判断AABB与圆形相交
/// </summary>
/// <param name="aABBArea"></param>
/// <param name="target"></param>
/// <returns></returns>
public static bool AABB(AABBArea aABBArea, CircleArea target)
{
Vector2 v = Vector2.Max(aABBArea.center - target.o, -(aABBArea.center - target.o));
Vector2 u = Vector2.Max(v - aABBArea.extents,Vector2.zero);
return u.sqrMagnitude < target.r * target.r;
}
圆形与OBB
定义OBB:
/// <summary>
/// OBB区域
/// </summary>
public struct OBBArea
{
public Vector2 center;
public Vector2 extents;
public float angle;
}
OBB相对于AABB,矩形边不与坐标轴重合,对于它和圆形的相交检测只需要把圆形旋转到OBB边所在坐标系中,剩下的步骤与AABB的相同。
相交检测:
/// <summary>
/// 判断OBB与圆形相交
/// </summary>
/// <param name="oBBArea"></param>
/// <param name="target"></param>
/// <returns></returns>
public static bool OBB(OBBArea oBBArea, CircleArea target)
{
Vector2 p = oBBArea.center - target.o;
p = Quaternion.AngleAxis(-oBBArea.angle, Vector3.forward) * p;
Vector2 v = Vector2.Max(p, -p);
Vector2 u = Vector2.Max(v - oBBArea.extents, Vector2.zero);
return u.sqrMagnitude < target.r * target.r;
}
来源:https://blog.csdn.net/qq_37043683/article/details/80375691


猜你喜欢
- 本文实例讲述了C#中实现一次执行多条带GO的sql语句。分享给大家供大家参考。具体如下:using System;using System.
- 本文实例讲述了C#使用ToUpper()与ToLower()方法将字符串进行大小写转换的方法。分享给大家供大家参考。具体分析如下:C#通过T
- 在进行java编程的时候,我们可以生成可运行的jar文件,但是鉴于平台的不同,我们可能需要将jar文件转化为exe格式。今天,小编就用一款叫
- Remote Debug 综述当我们的后台项目部署到服务器上时,由于环境和本地不同,有时候也会有一些奇奇怪怪的问题出现。只依赖服务器上的日志
- 数组的定义数组本质上就是让我们能 " 批量 " 创建相同类型的变量。数组的三种语法格式1、 数据类型 [] 数组名称 =
- 前言我们来分析一下堆内布局以及Java对象在内存中的布局吧。对象的指向先来看一段代码:package com.zwx.jvm;public
- 1 简介Solace是一个强大的实时性的事件驱动消息队列。本文将介绍如何在Spring中使用,虽然代码使用的是Spring Boot,但并没
- package com.weixin.util;import java.io.IOException;import java.util.Ra
- 一.OO(面向对象)的设计基础面向对象(OO):就是基于对象概念,以对象为中心,以类和继承为构造机制,充分利用接口和多态提供灵活性,来认识、
- MD5加密在我们的程序中,不管是什么,都会有安全问题,今天就说的是MD5加密的方法MD5是哈希算法,也就是 从明文A到密文B很容易,但是从密
- 前言本文主要给大家介绍了关于如何实现Builder模式,大家在构建大对象时,对象的属性比较多,我们可以采用一个构造器或者使用空的构造器构造,
- 一、介绍JUnit是一款优秀的开源Java单元测试框架,也是目前使用率最高最流行的测试框架,开发工具Eclipse和IDEA对JUnit都有
- 本文实例为大家分享了Android保存QQ密码功能的具体代码,供大家参考,具体内容如下技术要点:使用文件储存的方式保存数据实现步骤:①用户交
- 本文实例总结了C#实现按照指定长度在数字前补0方法。分享给大家供大家参考。具体分析如下:这里分析了C#按照指定的长度在数字前补0的两种方法例
- 下面通过代码给大家介绍c++ string insert() 函数,具体内容如下:basic_string& inser
- 在开发过程中,碰到生成一个List对象,需要对其里面的每个对象都进行校验。但是,这个Lis
- 话不多说,请看代码:using System;using System.Web;using System.Drawing;using Sys
- 基于SSM框架的仓库管理系统功能:系统操作权限管理。系统提供基本的登入登出功能,同时系统包含两个角色:系统超级管理员和普通管理员,超级管理员
- 这篇文章主要介绍了JAVA利用递归删除文件代码实例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可
- 学习大佬们开发安全小工具,打包jar解决错误: 找不到或无法加载主类 main1 Maven方式遇到报错”找不到或无法加载主类 main“解