Unity查找游戏物体的六种方式详解
作者:陈言必行 发布时间:2021-07-09 18:23:17
一篇小白也能看懂的查找游戏物体的方式解析 – Unity 之 查找物体的几种方式。本文通过实际测试得出使用结论,大家进行简单记录,在使用时想不起来可以再来看看,多用几次基本就没有问题了。
一,Object.Find()
Object.Find()
:根据名称找到游戏对象并返回它。
void ObjectFind()
{
// 找父级
GameObject parent = GameObject.Find("GameObject");
Debug.Log("找父级物体,是否找到:" + (parent != null));
// 找子级
GameObject child = GameObject.Find("Child");
Debug.Log("找子级物体,是否找到:" + (child != null));
// 找父级隐藏物体
GameObject parentHide = GameObject.Find("GameObjectHide");
Debug.Log("找父级隐藏物体,是否找到:" + (parentHide != null));
// 找子级隐藏物体
GameObject childHide = GameObject.Find("ChildHide");
Debug.Log("找子级隐藏物体,是否找到:" + (childHide != null));
}
测试结果如下图:
当有使用GameObject.Find("GameObject")
, 场景中有多个名为“GameObject”的物体存在时,将每个“GameObject”设置为不同的标签,多运行几次查看结果。
测试场景如下:
测试代码如下:
// 找同名物体
GameObject nameObj = GameObject.Find("GameObject");
Debug.Log("找同名,是否找到:" + nameObj.tag);
测试结果: 查找顺序是:“自身”(挂载脚本的物体) --> 和自身同层级上面物体 --> 和自身同层级下面物体 --> 自身子物体 --> 自身父物体。
Object.Find()得出结论:
全局查找参数名称游戏物体;
不对禁用(隐藏)物体进行查找;
若有同名物体时根据层级关系进行查找。
使用建议: 有同名物体存在时,尽量不要使用Object.Find()
进行查找,或者说使用Object.Find()
进行查找时,应控制查找物体命名唯一。
二,FindGameObjectWithTag()
GameObject.FindGameObjectWithTag()
根据标签查找游戏物体并返回。GameObject.FindGameObjectsWithTag()
根据标签查找当前场景中所有这个标签的游戏物体并返回所有物体的数组。
将如下场景:除主摄像机~(Main Camera)外的所有游戏物体的标签(Tag)~都修改为Player,进行测试。
测试代码如下:
void GameObjectFindWithTag()
{
GameObject tagObj = GameObject.FindGameObjectWithTag("MainCamera");
Debug.Log("根据标签查找游戏物体,是否查到:" + (tagObj != null));
GameObject[] tagObjs = GameObject.FindGameObjectsWithTag("Player");
for (int i = 0; i < tagObjs.Length; i++)
{
Debug.Log("根据标签查找游戏物体名称:" + tagObjs[i].name);
}
}
测试结果:
查找不存在的标签测试:
GameObject tagObj = GameObject.FindGameObjectWithTag("MainCamera1");
Debug.Log("根据标签查找游戏物体,是否查到:" + (tagObj != null));
报错:UnityException: Tag: MainCamera1 is not defined.
翻译: MainCamera1是一个未定义的标签
FindGameObjectWithTag()得出结论:
查找不到禁用物体,使用时需确认要查找的物体是启用(显示)状态;
有多个有游戏物体使用同一标签时,尽量不使用FindGameObjectWithTag此方式查找单一游戏体,因为查找顺序会受到层级影响;
查找未定义标签会报错,使用时需确认查找的字符串是已定义的标签;
查找的标签是已定义但是未使用过,会找不到游戏物体,返回空值。
三,GameObject.FindObjectOfType()
和上面根据标签查找的逻辑差不多。
GameObject.FindObjectOfType<类型>();
:根据类型(组件/自定义脚本)查找并返回这个类。
GameObject.FindObjectsOfType<类型>()
:根据类型(组件/自定义脚本)查找当前场景中所有这个类并返回一个这个类的数组。
void FindObjectOfType()
{
Camera typeCamera = GameObject.FindObjectOfType<Camera>();
Debug.Log("根据类型查找物体,是否查到:" + (typeCamera != null));
Transform[] typeTransArr = GameObject.FindObjectsOfType<Transform>();
for (int i = 0; i < typeTransArr.Length; i++)
{
Debug.Log("根据类型查找到的物体名称:" + typeTransArr[i].name);
}
}
FindObjectOfType()得出结论:
查找不到禁用物体,使用时需确认要查找的物体是启用(显示)状态;
查找场景中不存在类型时会返回null,不会报错;
通常使用情况为:初始化时在一个脚本中获取另一个脚本的引用,通过这种形式查找。【后多被单例取代】
四,Transform.Find()
查找挂载物体父级,同级,子级物体:
void TransformFind()
{
// 找父级
Transform parent = transform.Find("Root");
Debug.Log("找父级物体,是否找到:" + (parent != null));
// 找同级
Transform selfObj = transform.Find("Parent_1");
Debug.Log("找同级物体,是否找到:" + (selfObj != null));
// 找子级
Transform child = transform.Find("Child");
Debug.Log("找子级物体,是否找到:" + (child != null));
// 找子级隐藏物体
Transform childHide = transform.Find("ChildHide");
Debug.Log("找子级隐藏物体,是否找到:" + (childHide != null));
}
找多层级子物体:
// 找二级子物体
Transform child_1 = transform.Find("Child_1_1");
Debug.Log("找二级子物体 参数只写名称,是否找到:" + (child_1 != null));
// 找二级子物体
Transform child_1_1 = transform.Find("Child/Child_1_1");
Debug.Log("找二级子物体 参数写全路径,是否找到:" + (child_1_1 != null));
Find()得出结论:
只能找其子物体,不能找其同级或更高层级物体
找子物体时不考虑是否被禁用(隐藏)
找多层子物体时需写全路径(否则即使存在也找不到)
五,Transform.FindObjectOfType()
经过测试和GameObject.FindObjectOfType()
没什么区别,测试结果一致,测试代码和截图就不发处理占地方了。
测试时我发现 GameObject.FindObjectsOfType<类型>()
和Transform.FindObjectsOfType<Transform>()
被合并了,应该说完全是一个方法了,根据下图可以看到,我虽然前打的是Transform
的标签,但是它是灰色的,鼠标放上去看到方法引用的却是GameObject.FindObjectsOfType
。
得出结论:
Transform.FindObjectOfType() 和 GameObject.FindObjectOfType()使用方式一样,结果也没有区别…
六,transform.GetChild()
Transform.GetChild()
是找子物体的方法,也是我个人比较喜欢用的方式,弊端是不能随意修改游戏物体的层级关系。
使用起来也很简单
比如:找一级子物体的第一个物体
Transform child1 = transform.GetChild(0);
找一级子物体的第一个物体的第三个子物体
Transform child1 = transform.GetChild(0).GetChild(2);
使用方式:几个层级就几个GetChild()
,参数就是当前层级的第几个物体(从0开始)
使用拓展:
遍历子物体:
for (int i = 0; i < transform.childCount; i++)
{
Debug.Log(transform.transform);
}
获取当前物体的父物体transform.parent
获取当前物体的根物体transform.root
transform.GetChild()使用总结:
以自身为基础,查找子物体(注意索引从0开始,写多报错)
可以使用transform.parent.parent 的形式无限向上,然后再GetChild(),就达到了查找父层级或更高层级物体的目的
弊端是依赖游戏物体的层级关系,使用时需确保层级关系相对稳定。若不稳定会导致每次修改游戏体时还要修改代码,这就加大了工作量了。
总结
来源:https://blog.csdn.net/Czhenya/article/details/117530824
猜你喜欢
- 本文实例讲述了c#与js随机数生成方法。分享给大家供大家参考。具体如下:1. C#产生随机数方法:Random rd = new Rando
- 背景大家在使用Selenium + Chromedriver爬取网站信息的时候,以为这样就能做到不被网站的反爬虫机制发现。但是实际上很多参数
- 1、在build.gradle(Module)里引入依赖,然后重构(sync Now):android { ...
- 本文实例总结了C#中split用法。分享给大家供大家参考,具体如下:以下是我转载的两个不同的人的,方便大家及自己查阅string s=&qu
- 前言本文主要介绍的是关于CentOS 7配置自定义JDK的方法教程,分享出来供大家参考学习,下面来一起看看详细的介绍:配置教程由于选择的是C
- 一、实体类转换成XML将实体类转换成XML需要使用XmlSerializer类的Serialize方法,将实体类序列化public stat
- 本文较为深入的分析了android中UI主线程与子线程。分享给大家供大家参考。具体如下:在一个Android 程序开始运行的时候,会单独启动
- 前言在foreach中删除元素时,每一次删除都会导致集合的大小和元素索引值发生变化,从而导致在foreach中删除元素时会抛出异常。集合已修
- 多表联合查询resultType的返回值一般数据按参数类型返回<select id="queryCarIdList"
- Java对自动登陆功能的简单实现,仅用到了servlet和一个filter。第一次写博文,如有不足的地方,敬请指正,谢谢。登陆界面<%
- 摘要在J2SE 1.5的java.util.concurrent包(下称j.u.c包)中,大部分的同步器(例如锁,屏障等等)都是基于Abst
- 如下XML文件:(算是一个属性值比较多的xml文件。。。读取该Xml算是我在公司实际的一个任务)<?xml version="
- MultipartResolver和ServletFileUpload冲突如果同时使用了MultipartResolver 和Servlet
- 指定委托那些办理者处理任务节点,也就是只有此办理者才有操作此节点的操作权限1、直接指定流程图配置实现:在流程图属性栏设置Assignee属性
- spring-boot-devtools是一个为开发者服务的一个模块,其中最重要的功能就是自动应用代码更改到最新的App上面去。原理是在发现
- 前言继上次文章C# 初识Picture控件,今天我们来学一下PictureBox的方法以及一些基本操作,对于图片的删除,上传以及重命名,之前
- 引出泛型我们通过如下的示例,引出为什么泛型的概念。public class Test {public static void main(St
- 前言最近学习java,接触到了回调机制(CallBack)。初识时感觉比较混乱,而且在网上搜索到的相关的讲解,要么一言带过,要么说的比较单纯
- 前言这里介绍一个.net自身携带的类ImageAnimator,这个类类似于控制动画的时间轴,使用ImageAnimator.CanAnim
- 一般都在windows下开发的,现在部署到linux下:1,将项目达成war包(用eclipse,项目右键-->Export-->