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 脚本编程与开发》 如侵告删

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/158505.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

CSS网页标题图案和LOGO SEO优化

favicon图标 将网页的头名字旁边放入一个图案 想将想要的图案切成png图片 然后把png图片转换成ico图案可以借助进行访问 将语法引用到head里面 SEO译为搜索引擎优化。是一种利用搜索引擎的规则提高网站有关搜索引擎的自然排名的方式 SEO的目的是对网站进行深度的优化&…

Unity关键词语音识别

一、背景 最近使用unity开发语音交互内容的时候,遇到了这样的需求,就是需要使用语音关键字来唤醒应用程序,然后再和程序做交互,有点像智能音箱的意思。具体的技术方案方面,也找了一些第三方的服务,比如百度…

【Mybatis】基于Mybatis插件+注解,实现敏感数据自动加解密

一、介绍 业务场景中经常会遇到诸如用户手机号,身份证号,银行卡号,邮箱,地址,密码等等信息,属于敏感信息,需要保存在数据库中。而很多公司会会要求对数据库中的此类数据进行加密存储。 敏感数据…

C语言达到什么水平才能从事单片机工作

C语言达到什么水平才能从事单片机工作 从事单片机工作需要具备一定的C语言编程水平。以下是几个关键要点:基本C语言知识: 掌握C语言的基本语法、数据类型、运算符、流控制语句和函数等基本概念。最近很多小伙伴找我,说想要一些C语言学习资料&…

下载Python的不同版本在同一台电脑上如何共存

1. 下载安装不同版本的Python 官网下载:https://www.python.org/downloads/安装自己需要的版本(我这里以Python3.6和Python3.9为例,下载安装细节不过多赘述) (这里的安装路径自己设定,命名最好是根据下载…

win10取消ie浏览器自动跳转edge浏览器

建议大家看完整篇文章再作操作 随着windows10 日渐更新,各种不同的操作,规避IE浏览器跳转Edge浏览器的问题 算了,找了台云机装的server 有自带的IE 1.(失败)思路 协助Edge浏览器 管理员身份打开 PowerShell 一般e…

Arcgis实现Tiff合并

Arcgis实现Tiff合并 现有四幅Tiff影像 打开数据管理工具 输入使用这四幅影像 下面这个就是建立数据库,这个不对 点击确定 合成完毕

网络安全就业形势怎么样?

泻药,以下都是我本人的肺腑之言,是答主深耕职场多年,转战数家公司总结周围朋友的从业经验才总结出来的行业真相,真心希望帮助到还没有步入职场的大家,尤其是24届的应届毕业生,多掌握些就业信息就能少走一些…

采用 guidance 提高大模型输出的可靠性和稳定性

本文首发于博客 LLM 应用开发实践 在复杂的 LLM 应用开发中,特别涉及流程编排和多次 LLM 调用时,每次的 Prompt 设计都取决于前一个步骤的大模型输出。如何避免大语言模型的"胡说八道",以提高大语言模型输出的可靠性和稳定性&#…

时序数据库InfluxDB了解

参考:https://blog.csdn.net/u014265785/article/details/126951221

【C++】 局部对象,引用返回

1、new 关键字 会在堆内申请空间,如果仅仅是普通调用构造函数,不会在堆内开辟空间。 2、函数调用会形成栈帧,进行压栈操作,函数调用结束,会进行弹栈。 函数内的局部对象,会随着弹栈,而被销毁(…

Jetson Orin NX 开发指南(5): 安装 OpenCV 4.6.0 并配置 CUDA 以支持 GPU 加速

一、前言 Jetson 系列的开发板 CPU 性能不是很强,往往需要采用 GPU 加速的方式处理图像数据,因此本文主要介绍如何安装带有 GPU 加速的 OpenCV,其中 GPU 加速通过 CUDA 来实现。 参考博客 Ubuntu 20.04 配置 VINS-Fusion-gpu OpenCV 4.6.…

SpringBoot Redis 基础使用

目录 Redis能做什么: redis特点 redis优点 opsForValue opsForList opsForSet opsForZSet StringRedisTemplate与RedisTemplate区别点 redis是一个key-value。和Memcached类似,它支持存储的value类型相对更多,包括string(字符串)、li…

MS9708数模转换器可pin对pin兼容AD9708

MS9708/MS9710/MS9714 是一个 8-Bit/10-Bit/14-Bit 高速、低功耗 D/A 转换器。可pin对pin兼容AD9708。当采样速率达到 125MSPS 时,MS9708/MS9710/MS9714 也能提供优越的 AC 和 DC 性能。 MS9708/MS9710/MS9714 正常工作电压范围为2.7V ~ 5.5V,功耗低能很…

尿检设备“智能之眼”:维视智造推出MV-MC 系列医疗专用相机

​ 尿液分析是临床检验的基础常规项目,随着医疗设备的不断发展,尿液分析相关仪器的国产化和自动化程度也进一步提升。2022 年国内尿液分析市场的规模约为 28 亿元,激烈的竞争推动了尿检仪器自动化、智能化升级,在仪器中加入机器视…

软件工程与计算总结(十一)人机交互设计

目录 ​编辑 一.引例 二.目标 三.人类因素 1.精神模型 2.差异性 四.计算机因素 1.可视化设计 2.常见界面类型 五.人机交互设计的交互性 1.导航 2.反馈 3.设计原则 六.设计过程 1.基本过程 2.界面原型化 一.引例 无论软件功能多么出色,亦或内部的构造…

SpringBoot 过滤器filter当中的自定义异常捕获问题

需求描述:需要根据用户的请求路径拦截做权限控制: 但是这样做全局异常无法捕获 解决方案: 在filter当中引入HandlerExceptionResolver类,通过该类的resolveException方法抛出自定义异常: public class OpenInvokeFil…

计算机指令、机器码

目录 背景 在软硬件接口中,CPU 帮我们做了什么事? 从编译到汇编,代码怎么变成机器码? 解析指令和机器码 总结延伸 背景 上大学的时候,我们系里教 C 语言程序设计的老师说,他们当年学写程序的时候&…

实施运维03(在虚拟机上安装winServer2008系统)

新建虚拟机(一直下一步) 新建成功后选择镜像(右键设置,选择CD/DVD,选择使用IOS镜像文件,浏览选择2008镜像打开) 安装2008版本系统(一直下一步) 修改密码---设置密码 与电脑远程连…

接口测试文档

接口测试的总结文档 第一部分:主要从问题出发,引入接口测试的相关内容并与前端测试进行简单对比,总结两者之前的区别与联系。但该部分只交代了怎么做和如何做?并没有解释为什么要做? 第二部分:主要介绍为什…