• 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对象有多种创建方法,例如以下方法。

    1. // 创建从原点向上的射线
    2. Ray ray = new Ray(Vector3.zero, Vector3.up);
    3. // 获得当前鼠标指针在屏幕上的位置(单位是像素)
    4. Vector2 mousePos = Input.mousePosition;
    5. // 创建一条射线,起点是摄像机位置,方向指向鼠标指针所在的点(隐含了从屏幕到世界的坐标转换)
    6. Ray ray2 = Camera.main.ScreenPointToRay(mousePos);
    7. // 之后可以将ray或ray2发射出去,例如:
    8. Physics.Raycast(ray, 10000, LayerMask.GetMask("Default"));

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

    3.2.5 层和层遮罩

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

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

    某些读者可能会很好奇,“与某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()函数的重载可以获取到碰撞信息。 

    1. bool Raycast(Vector3 origin, Vector3 direction, out RaycastHit hitInfo, float
    2. maxDistance);
    3. bool Raycast(Vector3 origin, Vector3 direction, out RaycastHit hitInfo, float
    4. maxDistance, int layerMask);
    5. bool Raycast(Ray ray, out RaycastHit hitInfo, float maxDistance, int layerMask);

    1. private void TestRay()
    2. {
    3. // 声明变量,用于保存碰撞信息
    4. RaycastHit hitInfo;
    5. // 发射射线,起点是当前物体的位置,方向是世界前方
    6. if (Physics.Raycast(transform.position, Vector3.forward, out hitInfo))
    7. {
    8. // 如果确实碰到物体,会运行到这里。没碰到物体就不会
    9. // 获取碰撞点的坐标(世界坐标)
    10. Vector3 point = hitInfo.point;
    11. // 获取对方的碰撞体组件
    12. Collider coll = hitInfo.collider;
    13. // 获取对方的Transform组件
    14. Transform trans = hitInfo.transform;
    15. // 获取对方的物体名称
    16. string name = coll.gameObject.name;
    17. // 获取碰撞点的法线向量
    18. Vector3 normal = hitInfo.normal;
    19. }

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

    2. 其他形状的射线

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

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

     

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

    1. // 球形射线:
    2. bool SphereCast(Ray ray, float radius);
    3. bool SphereCast(Ray ray, float radius, out RaycastHit hitInfo);
    4. // 盒子射线:
    5. bool BoxCast(Vector3 center, Vector3 halfExtents, Vector3 direction);
    6. bool BoxCast(Vector3 center, Vector3 halfExtents, Vector3 direction, out
    7. RaycastHit hitInfo, Quaternion orientation);
    8. // 胶囊体射线:
    9. bool CapsuleCast(Vector3 point1, Vector3 point2, float radius, Vector3 direction);
    10. bool CapsuleCast(Vector3 point1, Vector3 point2, float radius, Vector3 direction,
    11. out RaycastHit hitInfo, float maxDistance);

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

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

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

    3. 穿过多个物体的射线

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

    1. RaycastHit[] RaycastAll(Ray ray, float maxDistance);
    2. RaycastHit[] RaycastAll(Vector3 origin, Vector3 direction, float maxDistance);
    3. RaycastHit[] RaycastAll(Ray ray, float maxDistance, int layerMask);
    4. RaycastHit[] RaycastAll(Ray ray);

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

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

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

    1. Collider[] OverlapBox(Vector3 center, Vector3 halfExtents, Quaternion
    2. orientation, int layerMask);
    3. Collider[] OverlapCapsule(Vector3 point0, Vector3 point1, float radius, int
    4. layerMask);
    5. Collider[] OverlapSphere(Vector3 position, float radius, int layerMask);

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

    5. 射线调试技巧

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

    1. void DrawLine(Vector3 start, Vector3 end, Color color);
    2. void DrawLine(Vector3 start, Vector3 end, Color color, float duration);
    3. void DrawRay(Vector3 start, Vector3 dir, Color color);
    4. void DrawRay(Vector3 start, Vector3 dir, Color color, float duration);

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

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

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

    1. // 以一个简单的射线为例
    2. Raycast(起点, 方向向量, 长度);
    3. // 对应的可视化线条
    4. DrawLine(起点, 起点+方向向量.normalized * 长度, Color.red);
    5. // 其中nomalized是将向量标准化,即方向不变长度变为1

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

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

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

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

  • 相关阅读:
    Plant Simulation 与Web交互 V3.0工具
    redo log、binlog的提问
    Leetcode 29. Divide Two Integers (Python)
    web server apache tomcat11-15-proxy
    Spring框架学习 -- 创建与使用
    GC 算法与种类
    时间模块之datatime模块、os模块、sys模块、json模块、json模块实操
    Mysql密码忘记修复
    Docker操作总结
    【MyBatis笔记12】MyBatis中二级缓存相关配置内容
  • 原文地址:https://blog.csdn.net/leoysq/article/details/133775570