Unity基础课程之物理引擎5-射线的使用方法总结

     在实际游戏开发时,不可避免地要用到各种射线检测。即便是一个不怎么用到物理系统的游戏,也很可能要用到射线检测机制。换句话说,射线检测在现代游戏开发中应用得非常广泛,超越了物理游戏的范围。下面简单举几个例子。

(1)游戏中有单击地面的操作,因此要发射射线以确定是否点中了可单击区域和单击位置的坐标。

(2)在判定子弹或技能是否击中目标时,如果采用碰撞体需要考虑子弹速度,且存在穿透问题,而射线是没有速度的(瞬时发生),不仅易于使用,而且综合效率更高。

(3)在3D动作游戏或2D动作游戏中,判断玩家是否落地时,可以向角色脚下发射射线;判断玩家是否接触墙壁时,可以往左右两侧发射射线;判断玩家是否需要低头时,可以往头顶发射射线;判断玩家是否需要攀爬时,同样也可以采用射线检测的方法。

(4)因为射线与视线一样会被障碍物阻挡,所以在游戏AI设计中,可以用射线模拟AI角色的视线

     

注意,上所述的各种射线检测都是以物理系统为基础的。射线需要与碰撞体和触发器配合才能发挥出作用

下面来介绍一下射线编程方法。常用的直线型射线用类型Ray表示。Ray包含了origin(起点)和direction(方向)的定义,起点和方向都用Vector3类型表示,前者是一个坐标,后者是一个表示方向的向量。有很多方法可以在游戏世界中发射一条射线,最常用的方法是Physics.Raycast()和Physics.RaycastAll()。由于实践中有各式各样的具体应用场景,因此Physics.Raycast()方法的重载有10种以上,不过实际大同小异,例如以下3种。

bool Raycast(Vector3 origin, Vector3 direction);

bool Raycast(Vector3 origin, Vector3 direction, float maxDistance);

bool Raycast(Vector3 origin, Vector3 direction, float maxDistance, int layerMask);

以上3个函数共同的参数都是发射点坐标和方向向量,返回值都是是否击中了某个碰撞体或触发器。第3个参数maxDistance的作用是指定射线的最大长度。虽然名字叫作“射线”,但与几何中的射线不同,这里的“射线”更多是“发射”的意思。例如游戏中经常通过往角色脚下发射很短的射线(0.01,代表1厘米)来判断角色是否站在地上。除了指定方向和位置的射线以外,以下还有一类很常用的重载形式。

bool Raycast(Ray ray, out RaycastHit hitInfo);

bool Raycast(Ray ray, out RaycastHit hitInfo, float maxDistance);

bool Raycast(Ray ray, out RaycastHit hitInfo, float maxDistance, int layerMask);

这种形式的射线检测用了一种常用结构体Ray(射线),它只是将射线数据对象先单独创建出来,并没有实际区别。Ray对象有多种创建方法,例如以下方法。

//  创建从原点向上的射线
Ray ray = new Ray(Vector3.zero, Vector3.up);

//  获得当前鼠标指针在屏幕上的位置(单位是像素)
Vector2 mousePos = Input.mousePosition;
//  创建一条射线,起点是摄像机位置,方向指向鼠标指针所在的点(隐含了从屏幕到世界的坐标转换)
Ray ray2 = Camera.main.ScreenPointToRay(mousePos);

//  之后可以将ray或ray2发射出去,例如:
Physics.Raycast(ray, 10000, LayerMask.GetMask("Default"));

 这些重载形式的第2个参数,即类型为RaycastHit的参数hitInfo也很有用,它保存着详细的碰撞信息,如碰撞点的配置、法线等。碰撞信息会在第3.2.6小节重点详细讲解。

3.2.5 层和层遮罩

很多时候,需要射线仅被某些物体阻挡,例如希望检测地面的射线只检测地面,而不要检测其他东西,也就是说应当穿过地面以外的东西。那么这里就要用到Layer和Layer Mask(层遮罩)的概念了。“层”的概念让物理系统变得更加好用和实用。例如一条子弹射线,仅让它碰到Ground(地面)、Player(玩家角色)和Obstacle(障碍物)这3个层,而不会和其他层的物体碰撞,其编写代码如下。

int mask = LayerMask.GetMask("Ground", "Player", "Obstacle");
if (Physics.Raycast(transform.position, Vector3.forward, mask))
{
    // 碰到了物体
}

某些读者可能会很好奇,“与某3层碰撞”这一条件竟然用一个int就能表示。这其实是一种二进制的妙用,用一个int最多可以表示32个层的遮罩,Layer和Tag最多也只有32个,这不是巧合。如果让mask表示这3层以外的所有层,则用一个二进制的取反运算即可,其方法如下。

mask = ~mask; // 英文波浪线,代表二进制取反

mask = ~mask; // 英文波浪线,代表二进制取反

有时需要改变物体所在的层,如将一个物体设置在Default层上,其方法如下。

gameObject.layer = LayerMask.NameToLayer("Default");

可以通过函数LayerMask.NameToLayer()将层名称转化为整数表示的层,也可以用函数LayerMask.LayerToName()将表示层的整数转化为层名字。

3.2.6 射线编程详解

1. 射线碰撞信息

前文举例的函数的返回值仅仅是“是否碰到了物体”,而无法确定碰撞点是哪里,也不知道碰到的物体是哪一个。射线检测其实有着丰富的碰撞信息,如可以获取到碰撞点坐标、被碰撞物体的所有信息,甚至可以获取到碰撞点的法线(碰撞点所在物体平面的朝向)。这些丰富的碰撞信息,都被保存在RaycastHit结构体中。例如,以下几个Raycast()函数的重载可以获取到碰撞信息。 

bool Raycast(Vector3 origin, Vector3 direction, out RaycastHit hitInfo, float 
maxDistance);
bool Raycast(Vector3 origin, Vector3 direction, out RaycastHit hitInfo, float 
maxDistance, int layerMask);
bool Raycast(Ray ray, out RaycastHit hitInfo, float maxDistance, int layerMask);

    private void TestRay()
    {
        // 声明变量,用于保存碰撞信息
        RaycastHit hitInfo;
        // 发射射线,起点是当前物体的位置,方向是世界前方
        if (Physics.Raycast(transform.position, Vector3.forward, out hitInfo))
        {
            //  如果确实碰到物体,会运行到这里。没碰到物体就不会

            //  获取碰撞点的坐标(世界坐标)
            Vector3 point = hitInfo.point;
            //  获取对方的碰撞体组件
            Collider coll = hitInfo.collider;
            //  获取对方的Transform组件
            Transform trans = hitInfo.transform;
            //  获取对方的物体名称
            string name = coll.gameObject.name;

            //  获取碰撞点的法线向量
            Vector3 normal = hitInfo.normal;
        }

 以上例子基本涵盖了能从hitInfo中获取到的信息,更多碰撞信息可以查阅Raycastlift结构体的定义。

2. 其他形状的射线

射线不仅可以有长度,还可以有粗细和形状。除了前面所提到的直线射线,还有球形射线、盒子射线和胶囊体射线,如图3-7所示。

图3-7 3种形状的射线示意图

 

与发射射线类似,各种形状的射线也有很多种函数重载,以下是几种常用的重载形式。

//  球形射线:
bool SphereCast(Ray ray, float radius);
bool SphereCast(Ray ray, float radius, out RaycastHit hitInfo);

//  盒子射线:
bool BoxCast(Vector3 center, Vector3 halfExtents, Vector3 direction);
bool BoxCast(Vector3 center, Vector3 halfExtents, Vector3 direction, out 
RaycastHit hitInfo, Quaternion orientation);

//  胶囊体射线:
bool CapsuleCast(Vector3 point1, Vector3 point2, float radius, Vector3 direction);
bool CapsuleCast(Vector3 point1, Vector3 point2, float radius, Vector3 direction, 
out RaycastHit hitInfo, float maxDistance);

 可以看出,球形射线、盒子射线和胶囊体射线的发射函数与直线型射线是类似的。区别在于,球形射线需要指定球的半径;

盒子射线需要指定盒子的中心点和盒子的半边长(边长的一半),如果有必要再加上盒子的朝向;胶囊体的形状更为复杂,需要用point1、point2和radius(半径)这3个参数指定胶囊体的起点和形状。

在实践中有各种不同的需求和情况,在必要时可以进一步查阅相关资料,并对参数的用法做实际的试验。本小节的最后还会介绍射线调试的一些技巧。

3. 穿过多个物体的射线

有时需要射线在遇到第一个物体时不停止,继续前进,最终穿过多个物体。使用Physics.RaycastAll()函数可以获取到射线沿途碰到的所有碰撞信息,该函数的返回值是RaycastHit数组。

RaycastHit[] RaycastAll(Ray ray, float maxDistance);
RaycastHit[] RaycastAll(Vector3 origin, Vector3 direction, float maxDistance);
RaycastHit[] RaycastAll(Ray ray, float maxDistance, int layerMask);
RaycastHit[] RaycastAll(Ray ray);

同样,也有球形穿越射线、盒子穿越射线和胶囊体穿越射线,函数名称分别为SpherecastAll、BoxcastAll和CapsulecastAll。

4. 区域覆盖型射线(Overlap)

有时需要检测一个空间范围,例如炸弹爆炸时,范围10米之内的物体都会受到波及,那么这里需要的就不是一条射线,而是一个半径为10米的球形区域。物理系统也提供了这类函数,它们均以Physics.Overlap开头,列举如下。

Collider[] OverlapBox(Vector3 center, Vector3 halfExtents, Quaternion 
orientation, int layerMask);
Collider[] OverlapCapsule(Vector3 point0, Vector3 point1, float radius, int 
layerMask);
Collider[] OverlapSphere(Vector3 position, float radius, int layerMask);

以球形覆盖检测OverlapSphere()为例,调用该函数时,会返回原点为position、半径为radius的球体内,满足一定条件的碰撞体集合(以数组表示),而这个球体称为“3D相交球”。

5. 射线调试技巧

射线检测函数类型多、重载多、参数多,可能会让读者看得一头雾水。在实际游戏开发中,虽然这些参数不容易填写正确,但也有很好的方法可以提高编程的效率。这个方法就是使用Debug.DrawLine()函数和Debug.DrawRay()函数,将看不见的射线以可视化的形式表现出来,方便查看参数是否正确。Debug.DrawLine()函数和Debug.DrawRay()函数的常用形式如下。

void DrawLine(Vector3 start, Vector3 end, Color color);
void DrawLine(Vector3 start, Vector3 end, Color color, float duration);

void DrawRay(Vector3 start, Vector3 dir, Color color);
void DrawRay(Vector3 start, Vector3 dir, Color color, float duration);

 Debug.DrawLine()函数通过指定线段的起点、终点和颜色(默认红色),绘制一条线段;

Debug.DrawRay函数则是通过指定起点和方向向量,绘制一条射线。

两者的用法是相似的。使用时要注意,发射射线时,参数通常为起点、方向向量和长度,而DrawLine()方法用的是起点和终点。应正确使用向量加法,避免看到的线条与实际射线不一致。下面举个例子以供读者参考。

//  以一个简单的射线为例
Raycast(起点, 方向向量, 长度);

//  对应的可视化线条
DrawLine(起点, 起点+方向向量.normalized * 长度, Color.red);
//  其中nomalized是将向量标准化,即方向不变长度变为1

需要说明的是,这种绘制方法仅在开发期生效,不会出现在最终的游戏发布版中。在默认情况下,该辅助线仅在编辑器的场景窗口中可见。

如果要在Game窗口中看到它,则需要单击Game窗口右上角的Gizmos(辅助线框)按钮,而且无论怎么设置,它都不会出现在最终的游戏发布版中。

以上函数的最后一个参数,即持续时间(duration)可以省略,省略后这条参考线只出现一帧。如果在代码中每帧都绘制线条,那么就可以省略该参数。如果这个线条只出现一帧且看不清,则可以填写一个较大的持续时间(单位是秒),让射线停留在屏幕上方以便查看。

以上内容来源于《Unity3d 脚本编程与开发》 如侵告删

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Unity3d青子

难题的解决使成本节约,求打赏

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值