Unity基础5——物理检测

这篇具有很好参考价值的文章主要介绍了Unity基础5——物理检测。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

一、层级 Layer

​ Unity 中设置了共 32 层 Layer,如图,可以点击 Add Layer 添加自定义的 Layer

Unity基础5——物理检测

​ 通过名字得到层级编号 LayerMask.NameToLayer(string layer)
​ 我们需要通过编号左移构建二进制数,这样每一个编号的层级都是对应位为 1 的 2 进制数
​ 我们通过位运算可以选择想要检测层级,使用一个 int 就可以表示所有想要检测的层级信息

​ 层级编号是 0 ~ 31,刚好 32 位,是一个 int 数
​ 每一个编号代表的都是二进制的一位,例如
0 — — 1 < < 0 — — 0000 0000 0000 0000 0000 0000 0000 0001 = 1 1 — — 1 < < 1 — — 0000 0000 0000 0000 0000 0000 0000 0010 = 2 2 — — 1 < < 2 — — 0000 0000 0000 0000 0000 0000 0000 0100 = 4 3 — — 1 < < 3 — — 0000 0000 0000 0000 0000 0000 0000 1000 = 8 4 — — 1 < < 4 — — 0000 0000 0000 0000 0000 0000 0001 0000 = 16 5 — — 1 < < 5 — — 0000 0000 0000 0000 0000 0000 0010 0000 = 32

二、范围检测

​ 游戏中瞬时的攻击范围判断一般会使用范围检测
​ 举例:

  • 玩家在前方 5m 处释放一个地刺魔法,在此处范围内的对象将受到地刺伤害
  • 玩家攻击,在前方 1 米圆形范围内对象都受到伤害

​ 等等
​ 类似这种并没有实体物体,只想要检测在指定某一范围是否让敌方受到伤害时,便可以使用范围判断
​ 简而言之,在指定位置进行范围判断,我们可以得到处于指定范围内的对象,目的是对对象进行处理
​ 比如受伤、减血等等

​ 必备条件:想要被范围检测到的对象,必须具备碰撞器 Collider
​ 注意点:

  • 范围检测相关 API,只有当执行该句代码时进行一次范围检测,它是瞬时的
  • 范围检测相关 API,并不会真正产生一个碰撞器,只是碰撞判断计算而已

(一)盒状范围检测

// 1.盒状范围检测
// 参数一:立方体中心点
// 参数二:立方体三边大小
// 参数三:立方体角度
// 参数四:检测指定层级(不填检测所有层)
// 参数五:是否忽略触发器 不填使用UseGlobal
//        UseGlobal-使用全局设置 
//        Collide-检测触发器 
//        Ignore-忽略触发器 
// 返回值:在该范围内的触发器(得到了对象触发器就可以得到对象的所有信息)
print(LayerMask.NameToLayer("UI"));
Collider[] colliders = Physics.OverlapBox(Vector3.zero, Vector3.one, Quaternion.AngleAxis(45, Vector3.up),
                                          (1 << LayerMask.NameToLayer("UI")) | (1 << LayerMask.NameToLayer("Default")),
                                          QueryTriggerInteraction.UseGlobal);

// 另一个API 将碰撞器存入给定的数组中
// 返回值:碰撞到的碰撞器数量
// 参数:传入一个数组进行存储
// Physics.OverlapBoxNonAlloc()
if (Physics.OverlapBoxNonAlloc(Vector3.zero, Vector3.one, colliders) != 0) { }

​ 其中,参数五的 UseGlobal 在 Eidt / Project Settings / Physics 中设置

Unity基础5——物理检测

(二)球形范围检测

// 2.球形范围检测
// 参数一:中心点
// 参数二:球半径
// 参数三:检测指定层级(不填检测所有层)
// 参数四:是否忽略触发器 不填使用UseGlobal
//        UseGlobal-使用全局设置 
//        Collide-检测触发器 
//        Ignore-忽略触发器 
// 返回值:在该范围内的触发器(得到了对象触发器就可以得到对象的所有信息)
colliders = Physics.OverlapSphere(Vector3.zero, 5, 1 << LayerMask.NameToLayer("Default"));

// 另一个API 
// 返回值:碰撞到的碰撞器数量
// 参数:传入一个数组进行存储
// Physics.OverlapSphereNonAlloc
if (Physics.OverlapSphereNonAlloc(Vector3.zero, 5, colliders) != 0) { }

(三)胶囊范围检测

// 3.胶囊范围检测
// 参数一:半圆一中心点
// 参数二:半圆二中心点
// 参数三:半圆半径
// 参数四:检测指定层级(不填检测所有层)
// 参数五:是否忽略触发器 不填使用UseGlobal
//        UseGlobal-使用全局设置 
//        Collide-检测触发器 
//        Ignore-忽略触发器 
// 返回值:在该范围内的触发器(得到了对象触发器就可以得到对象的所有信息)
colliders = Physics.OverlapCapsule(Vector3.zero, Vector3.up, 1, 1 << LayerMask.NameToLayer("UI"),
                                   QueryTriggerInteraction.UseGlobal);

// 另一个API 
// 返回值:碰撞到的碰撞器数量
// 参数:传入一个数组进行存储
// Physics.OverlapCapsuleNonAlloc
if (Physics.OverlapCapsuleNonAlloc(Vector3.zero, Vector3.up, 1, colliders) != 0) { }

三、射线检测

​ 例如以下的碰撞检测:

  1. 鼠标选择场景上一物体

  2. FPS 射击游戏(无弹道-不产生实际的子弹对象进行移动)

​ 等等,需要判断一条线和物体的碰撞情况如何实现呢?

​ 射线检测就是来解决这些问题的,它可以在指定点发射一个指定方向的射线,判断该射线与哪些碰撞器相交,得到对应对象

(一)射线

// 参数一:起点
// 参数二:方向(一定记住 不是两点决定射线方向,第二个参数 直接就代表方向向量)
Ray r = new Ray(Vector3.right, Vector3.forward);
// 目前只是申明了一个射线对象 对于我们来说 没有任何的用处

// Ray中的参数
print(r.origin);     // 起点
print(r.direction);  // 方向

// 摄像机发射出的射线
// 得到一条从屏幕位置作为起点
// 摄像机视口方向为 方向的射线
Ray r2 = Camera.main.ScreenPointToRay(Input.mousePosition);

​ 注意:

  • 单独的射线对于我们来说没有实际的意义
  • 我们需要用它结合物理系统进行射线碰撞判断

(二)碰撞检测

​ Physics 类中提供了很多进行射线检测的静态函数,他们有很多种重载类型

​ 我们只需要掌握核心的几个函数,其它函数自然就明白什么意思了
​ 注意:文章来源地址https://www.toymoban.com/news/detail-496761.html

  • 射线检测也是瞬时的
  • 执行代码时进行一次射线检测

1.最原始的射线检测

// 准备一条射线
Ray r3 = new Ray(Vector3.zero, Vector3.forward);
// 进行射线检测 如果碰撞到对象 返回true
// 参数一:射线
// 参数二: 检测的最大距离 超出这个距离不检测
// 参数三:检测指定层级(不填检测所有层)
// 参数四:是否忽略触发器 不填使用UseGlobal
//        UseGlobal-使用全局设置 
//        Collide-检测触发器 
//        Ignore-忽略触发器 
// 返回值:bool 当碰撞到对象时 返回 true 没有 返回false
if (Physics.Raycast(r3, 1000, 1 << LayerMask.NameToLayer("Monster"), QueryTriggerInteraction.UseGlobal))
    print("碰撞到了对象");

// 还有一种重载 不用传入 射线 直接传入起点 和 方向 也可以用于判断
// 就是把 第一个参数射线 变成了 射线的 两个点 一个起点 一个方向
if (Physics.Raycast(Vector3.zero, Vector3.forward, 1000, 1 << LayerMask.NameToLayer("Monster"),
                    QueryTriggerInteraction.UseGlobal)) print("碰撞到了对象2");

2.获取相交的单个物体信息

// 物体信息类 RaycastHit
RaycastHit hitInfo;
// 参数一:射线
// 参数二:RaycastHit是结构体 是值类型 Unity会通过out 关键字 在函数内部处理后 得到碰撞数据后返回到该参数中
// 参数三:距离
// 参数四:检测指定层级(不填检测所有层)
// 参数五:是否忽略触发器 不填使用UseGlobal
//        UseGlobal-使用全局设置 
//        Collide-检测触发器 
//        Ignore-忽略触发器 
if (Physics.Raycast(r3, out hitInfo, 1000, 1 << LayerMask.NameToLayer("Monster"),
                    QueryTriggerInteraction.UseGlobal)) {
    print("碰撞到了物体 得到了信息");

    // 碰撞器信息
    print("碰撞到物体的名字" + hitInfo.collider.gameObject.name);
    // 碰撞到的点
    print(hitInfo.point);
    // 法线信息
    print(hitInfo.normal);

    // 得到碰撞到对象的位置
    print(hitInfo.transform.position);

    // 得到碰撞到对象 离自己的距离
    print(hitInfo.distance);

    // RaycastHit 该类 对于我们的意义
    // 它不仅可以得到我们碰撞到的对象信息
    // 还可以得到一些 碰撞的点 距离 法线等等的信息
}

// 还有一种重载 不用传入 射线 直接传入起点 和 方向 也可以用于判断
if (Physics.Raycast(Vector3.zero, Vector3.forward, out hitInfo, 1000, 1 << LayerMask.NameToLayer("Monster"),
                    QueryTriggerInteraction.UseGlobal)) { }

3.获取相交的多个物体

// 可以得到碰撞到的多个对象
// 如果没有 就是容量为0的数组
// 参数一:射线
// 参数二:距离
// 参数三:检测指定层级(不填检测所有层)
// 参数四:是否忽略触发器 不填使用UseGlobal
//        UseGlobal-使用全局设置 
//        Collide-检测触发器 
//        Ignore-忽略触发器 
RaycastHit[] hits = Physics.RaycastAll(r3, 1000, 1 << LayerMask.NameToLayer("Monster"),
                                       QueryTriggerInteraction.UseGlobal);
for (int i = 0; i < hits.Length; i++) print("碰到的所有物体 名字分别是" + hits[i].collider.gameObject.name);

// 还有一种重载 不用传入 射线 直接传入起点 和 方向 也可以用于判断
// 之前的参数一射线 通过两个点传入
hits = Physics.RaycastAll(Vector3.zero, Vector3.forward, 1000, 1 << LayerMask.NameToLayer("Monster"),
                          QueryTriggerInteraction.UseGlobal);

// 还有一种函数 返回的碰撞的数量 通过out得到数据
if (Physics.RaycastNonAlloc(r3, hits, 1000, 1 << LayerMask.NameToLayer("Monster"),
                            QueryTriggerInteraction.UseGlobal) > 0) { }

(三)注意问题

// 注意:
// 距离、层级两个参数 都是int类型
// 当我们传入参数时 一定要明确传入的参数代表的是距离还是层级
// 举例
// 这样写是错误的 因为第二个参数 代表的是距离 不是层级
if (Physics.Raycast(r3, 1 << LayerMask.NameToLayer("Monster"))) { }
              QueryTriggerInteraction.UseGlobal) > 0) { }

到了这里,关于Unity基础5——物理检测的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 2022-04-20 Unity入门7——物理系统之碰撞检测

    一、刚体 Rigid Body ​ 刚体利用体积(碰撞器 Collider)进行碰撞计算,模拟真实的碰撞效果,产生力的作用 ​ 碰撞产生的必要条件: 两个物体都有碰撞器 Collider 至少一个物体有刚体 Mass:质量 默认为千克,质量越大惯性越大 Drag:空气阻力 根据力移动对象时影响空气阻力大

    2023年04月09日
    浏览(38)
  • Unity UGUI的PhysicsRaycaster (物理射线检测)组件的介绍及使用

    PhysicsRaycaster是Unity UGUI中的一个组件,用于在UI元素上进行物理射线检测。它可以检测鼠标或触摸事件是否发生在UI元素上,并将事件传递给相应的UI元素。 PhysicsRaycaster通过发射一条射线来检测UI元素。当射线与UI元素相交时,PhysicsRaycaster会将事件传递给相应的UI元素。 Event

    2024年02月15日
    浏览(46)
  • Unity所以说是玩具, UGUI的PhysicsRaycaster (物理射线检测)组件的介绍及使用

    PhysicsRaycaster是Unity UGUI中的一个组件,用于在UI元素上进行物理射线检测。它可以检测鼠标或触摸事件是否发生在UI元素上,并将事件传递给相应的UI元素。 PhysicsRaycaster通过发射一条射线来检测UI元素。当射线与UI元素相交时,PhysicsRaycaster会将事件传递给相应的UI元素。 Event

    2024年02月22日
    浏览(42)
  • Unity基础课程之物理引擎6-关于物理材质的使用和理解

     每个物体都有着不同的摩擦力。光滑的冰面摩擦力很小,而地毯表面的摩擦力则很大。另外每种材料也有着不同的弹性,橡皮表面的弹性大,硬质地面的弹性小。在Unity中这些现象都符合日常的理念。虽然从原理上讲,物体的摩擦力和弹性有着更复杂的内涵,例如普通的钢板

    2024年02月07日
    浏览(46)
  • Unity UGUI的Physi会随机改变csRaycaster (物理射线检测)组件的介绍及使用

    这里只介绍liteflow的简单基础使用以及作者对liteflow进行可视化扩展的相关阐述 背景:对于拥有复杂业务逻辑的系统承载着核心业务逻辑,这些核心业务逻辑涉及内部逻辑运算,缓存操作,持久化操作,外部资源调取,内部其他系统RPC调用等等。项目几经易手,维护的成本就

    2024年02月20日
    浏览(67)
  • Unity UGUI的PhysicsRayc可变化、可扩展aster (物理射线检测)组件的介绍及使用

    最近工作中需要用到MongoDB的事务操作,因此参考了一些资料封装了一个小的组件,提供基础的CRUD Repository基类 和 UnitOfWork工作单元模式。今天,就来简单介绍一下这个小组件。 MongoDB在4.2版本开始全面支持了多文档事务,至今已过了四年了,虽然我们可能没有在项目中用Mon

    2024年02月20日
    浏览(48)
  • Unity UG算法能力可视化UI的PhysicsRaycaster (物理射线检测)组件的介绍及使用

    PhysicsRaycaster是Unity UGUI中的一个组件,用于在UI元素上进行物理射线检测。它可以检测鼠标或触摸事件是否发生在UI元素上,并将事件传递给相应的UI元素。 PhysicsRaycaster通过发射一条射线来检测UI元素。当射线与UI元素相交时,PhysicsRaycaster会将事件传递给相应的UI元素。 Event

    2024年01月20日
    浏览(56)
  • Unity UGUI的Physics2DRaycaster (2D物理射线检测)组件的介绍及使用

    Physics2DRaycaster是Unity中的一个UGUI组件,用于在2D场景中进行物理射线检测。它可以检测鼠标或触摸事件在UI元素上的碰撞,并将事件传递给相应的UI元素。 Physics2DRaycaster通过发射一条射线来检测UI元素的碰撞。当射线与UI元素相交时,Physics2DRaycaster会将事件传递给相应的UI元素,

    2024年02月15日
    浏览(48)
  • unity06 脚本 世界物理坐标体系与脚本基础应用

    子节点相对父节点坐标,以父节点中心点为标准 Global,即世界坐标系 Local,即本地坐标系 以物体自身为轴 6个方向代表:上下 前后 左右 移动物体并不是匀速的 每次运动0.01米,但是间隔的deltaTime不固定 每秒移动3米就是3个格子 transform.Translate() 可实现物体的运动 transform.Tr

    2024年02月16日
    浏览(35)
  • Unity UGUI的PhysicsRaycaster (物理射相当于利用泛型函数保存了类型信息线检测)组件的介绍及使用

    这是在Datadog公司任职的Kevin Gosse大佬使用C#编写.NET分析器的系列文章之一,在国内只有很少很少的人了解和研究.NET分析器,它常被用于APM(应用性能诊断)、IDE、诊断工具中,比如Datadog的APM,Visual Studio的分析器以及Rider和Reshaper等等。之前只能使用C++编写,自从.NET NativeAOT发

    2024年01月22日
    浏览(64)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包