☺ unity的官网文档:脚本 - Unity 手册
■ 学习方式:
首先了解unity相关概述,快速认识unity编辑器,然后抓住重点的学:游戏对象、组件|C#脚本、预制体、UI
☺ 学习过程你会发现,其实Unity中主要是用c#进行开发。
因为在这个过程中,无非就是,对游戏对象通过挂载的C#脚本,修改一下组件的一些属性,控制一下激活之类的操作。到了UGUI,同样,也是通过挂载的C#脚本,监听到事件发生后,调用一下C#脚本定义的方法处理一下事件。
- unity快速上手文档:Setting up 3D Beginner - Unity Learn
- 视频[这种动手类的快速上手,建议跟着视频动手操作一番]:【Unity教程】零基础带你从小白到超神_哔哩哔哩_bilibili
- ★重点学习内容:游戏对象、组件|C#脚本、预制体、UI[GUI]
■ 开发工具:
unity、visual studio
1、Unity 概述
(1) 什么是Unity
Unity是一套具有完善体系与编辑器的跨平台游戏开发工具,也可以称之为游戏引擎游戏引擎是指一些编写好的可重复利用的代码与开发游戏所用的各功能编辑器Unity目前已超过50%的游戏引擎市场占有率。
游戏引擎=可复用代码+各种开发工具
(2) Unity引擎优势
1.基于C#编程,拥有易上手、高安全性
的特性
2.独特的面向组件
游戏开发思想让游戏开发更加简单易复用
3.拥有十分成熟的所见即所得开发编辑器
4.拥有良好生态圈
,商城中包含大量成熟的功能脚本与资源
5.强大的跨平台特性,可以制作PC、主机、手机、AR、VR等多平台游戏
2、Unity Hub
管理Unity项目,当电脑中有多个unity不同版本的项目的时候,管理就比较麻烦,可以通过unity hub 对各个不同的unity hub项目管理。
3、认识unity编辑器
(1) unity恢复默认窗口: Windows-> Layouts-> Default
(2) 项目设置
- 包括:物理、声音、质量、编辑器设置等等
(3) 全局设置
- 编辑器的常用设置
- 扩展工具,设置一下脚本使用visual studio 打开
(4) unity菜单栏介绍
■ 文件 File:文件的常规操作[项目/场景创建、保存等,还有打包项目]
■ 编辑 Edit:复制、剪切、粘贴等操作
■ 资源 Assets:创建[脚本、场景、动画等等]、导入导出等等
■ 游戏对象 GameObject:会在游戏场景中真正显示出来
■ 组件 Component:非常重要,因为unity是面向组件的
- 组件和游戏对象是不可分隔的关系,一个游戏对象身上可以挂多个组件
■ 2d 和 3d 场景的切换:
- 点一下2D,就变成了3D场景
(5) unity 项目的运行,是运行的是整个场景,而非单个的游戏对象
- 运行的单位是场景!
4、对游戏对象的常见操作
(0) transform 属性相关的操作的工具栏
(1) 拉伸
- 重置为000
(2) 隐藏或显示图层中的游戏对象
(3) 看到游戏对象的网格形状
- 为了方便观察,一般选择 Shaded Wireframe
(4) 选择游戏对象:直接鼠标点击一下即可
(5) 右键-眼睛环绕
(6) 手的模式下,结合鼠标滚轮,可以进行拖拽,调远调近观察
-
进入手的模式,快捷方式,直接按住鼠标滚轮
-
和右键-眼睛环绕一样,都是为了更好地全面观察游戏对象
(7) 父子游戏对象,父游戏对象的坐标改变会一起改变子游戏对象的坐标
- 一个游戏对象,创建后的默认坐标位置 position,默认是相对于世界坐标的位置,简称为世界坐标位置
- 如果一个游戏对象,变成了另外一个游戏对象的子游戏对象,那么它的坐标就是相对位置-【相对于父游戏对象的位置】,简称相对位置
- 最外层的游戏对象
- 对于父子游戏对象的坐标轴位置,父坐标轴位置默认在父子游戏对象的中间:
- 可以设置一下为轴心,父坐标位置即可回到父游戏对象身上
- 坐标轴,全局是针对世界坐标
(8) unity 打包操作
-
在unity 中将图片、模型文件[后缀是.fbx]等素材进行打包
-
打包完的文件类型是 Unity package file
(9) 标签和图层
- 标签:一般是咱自己用于在代码中,判断游戏对象是否是咱想要的游戏对象
- 图层:是unity做的一种归类划分,归类之后,可以给摄像机等做过滤显示;比如摄像机对于这个图层的是不显示的,还有对于碰撞,这个图层可以不去碰撞等等。
5、组件
(1) 什么是组件
组件就是功能,需要给游戏对象添加什么功能就只需给它添加上组件。
在Unity中,游戏物体是不具备任何功能的,如果想要为其添加功能,那么就需要为它添加该功能的组件,而每一个组件其实就是一个引擎内部的组件脚本
或是由自己编写的组件脚本
。也就是说,一个游戏物体(GameObject)会包含多个组件(Component),每一个组件又是一个组件脚本。
- 比如:给游戏对象添加上重力组件,该游戏对象就会往下掉。
- 又比如:transform 也是一个组件。
- 又比如:灯光之所以是灯光,那是因为它比别人多了一个灯光的组件。游戏对象其实都是默认自带了一些组件,比如立方体自带上了立方体的组件,球自带上了球的组件。
- 一个游戏对象(图层外层)对应一个脚本,当需要的功能[组件]不存在的时候,需要自己手动编写脚本!
(2) 手动编写C#脚本
- 在项目中手动添加上C#脚本
- 添加上脚本
每个游戏对象之所以表现形式不同,是因为绑定到他们身上的组件脚本不同!
(3) 组件的生命周期
- 从 Awake 到 OnDestroy:
- Awake 之后就是激活OnEnale,然后是开始Start
- 改变:按帧改变Update-LateUpdate,按固定时间频率改变FixedUpdate
- 取消激活OnDisable:和激活OnEnable是一对
- 最后如果删除了组件,则会执行销毁OnDestroy
▪ 激活OnEnale和取消激活OnDisable:
■ 脚本的生命周期的执行顺序
-
如果有多个脚本,test1.cs、test2.cs、test3.cs
-
脚本的执行顺序是:会想把test1.cs、test2.cs、test3.cs 的所有Awake方法执行完[如果有手动写Awake方法的情况!],接着是从test1.cs到test2.cs最后到test3.cs 执行下一个生命周期-激活OnEanale[如果有手动写OnEanale方法的情况!],然后又是从从test1.cs到test2.cs最后到test3.cs 执行下一个生命周期Start......
▷ 现阶段的生命周期从Awake开始,每次都是从项目中第一个脚本到最后一个脚本,执行。
▷ 接着进入下一个生命周期,所有的脚本也是从项目中第一个脚本到最后一个脚本,执行当前的生命周期!
-
如果想要更改项目中脚本之间的生命周期的执行顺序,可以在编辑->项目设置->脚本执行顺序设置一下
- 顺序添加错了,直接拖拽更改即可
6、预制体和变体
(1) 什么事预制体
其实,就是把场景中做好的东西[独立的游戏对象]封装成预制体[生成一个文件]。
- 预制体的后缀是 .prefab
(2) 预制体好处:
(3) 预制体和场景的游戏对象的区别:
① 直接修改预制体,则场景中的游戏对象会同步发生改变。但是如果只直接修改了场景中的游戏对象,预制体不变。
- 包括了直接在预制体身上挂着组件,场景中的游戏对象会同步挂载。
② 想要实现修改场景中的游戏对象,预制体同步发生改变:
- 选择“打开”模式
③ 场景中的游戏对象,挂载了组件,想要同步挂载到预制体身上:
- 单个挂载:
- 批量挂载:override批量应用到预制体
(4) 通过场景中的游戏对象,快速选中预制体
(5) 变体
预制件变体:对当前的游戏对象,生成一个变体,变体会依赖旧的预制体。
当旧的预制体发生改变,变体也会跟着一起改变。但是变体改变,预制体不会发生改变。
7、动态修改游戏对象的属性
(1) 游戏对象和所挂载的C#脚本之间的关系:
(2) C#脚本的类中添加了一个属性对象,将该属性对象和Unity的游戏对象进行关联
(3) 通过预制体,在场景中实例化出游戏对象
- 结果1:
- 结果2:
■ EmptyTest.cs 内容如下:
<span style="color:#333333"><span style="background-color:#ffffff"><code class="language-c language-c#">using System.Collections;
using System.Collections.Generic;
using UnityEngine;
<span style="color:#008000">/**
* EmptyTest 类是绑定在场景SamepleScene下的游戏对象Empty身上
* 补充,课外小知识,C#的 Debug.Log 方法,其实就是相当于 前端项目的 console.log 方法,也就是开发过程中用于打印日志的
*/</span>
public <span style="color:#0000ff">class</span> <span style="color:#a31515">EmptyTest</span> : MonoBehaviour
{
public GameObject Cube;
<span style="color:#008000">//预制体</span>
public GameObject Prefab;
<span style="color:#008000">// Start is called before the first frame update</span>
<span style="color:#a31515">void</span> <span style="color:#a31515">Start</span>()
{
<span style="color:#008000">//当前类的属性gameObejct就对应了unity当前脚本所绑定的游戏对象</span>
GameObject gameObject = this.gameObject;
Debug.Log(<span style="color:#a31515">"gameObject: "</span> + gameObject);
Debug.Log(<span style="color:#a31515">"gameObject的名称:"</span> + gameObject.name);
Debug.Log(<span style="color:#a31515">"gameObject的标签:"</span> + gameObject.tag);
Debug.Log(<span style="color:#a31515">"gameObject的图层:"</span> + gameObject.layer);
Debug.Log(<span style="color:#a31515">"Empty下的Cude的name:"</span> + Cube.name);<span style="color:#008000">//子物体的名称</span>
Debug.Log(<span style="color:#a31515">"Empty下的Cude的继承的父类的激活状态:"</span> + Cube.activeInHierarchy);<span style="color:#008000">//子物体继承关系的激活状态【看所继承的父物体的激活状态】</span>
Debug.Log(<span style="color:#a31515">"Empty下的Cude的自身的激活状态:"</span> + Cube.activeSelf);<span style="color:#008000">//子物体自身的激活状态</span>
Debug.Log(<span style="color:#a31515">"Empty的transform:"</span> + transform.position);<span style="color:#008000">//位置</span>
Debug.Log(<span style="color:#a31515">"Empty下的Cude的transform:"</span> + transform.Find(Cube.name).position);<span style="color:#008000">//位置</span>
<span style="color:#008000">//给游戏对象添加上组件,通过拿到游戏对象.AddComponent方法</span>
gameObject.AddComponent<MeshFilter>();
Cube.AddComponent<AudioSource>();
<span style="color:#008000">//获取其他组件【这个其他组件,其实就是排除掉Transform这个必须的组件后的其他组件】</span>
Debug.Log(<span style="color:#a31515">"Empty获取其他组件:"</span> + GetComponent<BoxCollider>());
Debug.Log(<span style="color:#a31515">"Cube获取其他组件:"</span> + Cube.GetComponent<BoxCollider>());
<span style="color:#008000">//从子物体身上获取组件</span>
<span style="color:#008000">// GetComponentInChildren<BoxCollider>();</span>
<span style="color:#008000">//从父物体身上获取组件</span>
<span style="color:#008000">//GetComponentInParent<BoxCollider>();</span>
<span style="color:#008000">//全局查找--细节,就是首字母大写了,小写的是属性</span>
<span style="color:#008000">//所谓的全局查找,其实就是对当前的C#脚本所处的场景SamepleScene下进行全局查找 </span>
GameObject testGameObject = GameObject.Find(<span style="color:#a31515">"TestGameObject"</span>);
<span style="color:#008000">//还可以通过标签名获取到游戏对象</span>
GameObject testGameObejct2 = GameObject.FindWithTag(<span style="color:#a31515">"enemy"</span>);
Debug.Log(<span style="color:#a31515">"testGameObject:"</span> + testGameObject);
Debug.Log(<span style="color:#a31515">"testGameObject2:"</span> + testGameObejct2);
<span style="color:#008000">//还可以设置游戏对象是否激活</span>
testGameObject.SetActive(<span style="color:#a31515">false</span>);<span style="color:#008000">//取消激活</span>
<span style="color:#008000">//需求:通过预制体,在场景中实例化出游戏对象</span>
<span style="color:#008000">//预制体的类型也是GameObject</span>
<span style="color:#008000">// Instantiate(Prefab);</span>
<span style="color:#008000">// 实例化出游戏对象,并且是当前挂载了C#脚本的游戏对象的子物体</span>
GameObject capsule = Instantiate(Prefab, transform);
<span style="color:#008000">// 销毁游戏对象</span>
Destroy(capsule);
}
<span style="color:#008000">// Update is called once per frame</span>
<span style="color:#a31515">void</span> <span style="color:#a31515">Update</span>()
{
}
}
</code></span></span>
8、Application 读写文件需要用到的类
(1) Application 读写文件
(2) Application 还有控制权限的作用,比如控制游戏后台运行,打开链接,退出游戏
9、游戏场景
■ 核心类-场景管理类 SceneManager:和场景的创建、场景切换|加载、场景的获取、场景数量、场景卸载有关
■ 关于场景的常用属性:场景名称、场景是否已加载、场景路径、场景索引、场景中独立的游戏对象
■ 异步加载场景并获取进度
- 通过协程的方式[协程,其实就是协助类型的多线程] StartCoroutine(任务);
- 在Update中获取到进度,因为Update是输出每一帧的改变
▷ 通过计时器控制场景跳转:
10、transform
(1) 位置、旋转、缩放、向量
(2) transform 其他关系
(2) transform 父子关系
游戏对象的父子级关系,其实是通过transform 维持的
......
11、Unity GUI(简称UGUI)
GUI:Graphical User Interface 图形用户界面或图形用户接口。
GUI,是平面的,在2d下操作
- UI 常用的控件
-
UI 最基本的是
画布Canvas
,相当于屏幕。- 最基本的UI就是画布Canvas,其他UI控件-Text、Image、Button等等都是放在画布上面的,作为Canvas的子控件。
-
UI 另外最基本的是
事件系统EventSystem
,其实就是把很多常见的事件,比如点击事件,封装起来。
(1) Canvas,相当于屏幕
■ Canvas组件的渲染模式一般默认是屏幕空间-覆盖
- 摄像机是拍摄到真正的游戏画面,3d画面。摄像机拍摄到画面会先被渲染出来。
- UI 覆盖:会覆盖到摄像机拍摄的画面之上。所以,UI 永远可以在最上层显示。
- 这也符合大多数真实的游戏情况,UI 显示在游戏画面之上。比如,打开一个背包界面或者人物属性界面,不可能被游戏场景的物体,比如被树木遮盖住。
■ Canvas组件的缩放模式一般是选择按屏幕分辨率缩放
- Canvas 做了适配,要真正生效,Canvas 下的第一层子控件也是要做适配。
-
第一层的子控件做适配:
-
锚点是设置在父物体身上的。
-
控件锚点的作用:控件永远会和锚点保持一定的距离。
-
(2) 事件绑定:以Button这个UI为例
① 在画布Canvas 下添加UI-Button,在项目下创建一个C#脚本,然后把C#脚本挂载到Button身上
② 接着在C# 方法,写一个点击事件发生时的处理方法 BtnClick()
③ 最后在Button 组件下面的On Click() 添加上C# 脚本中,点击事件发生时的处理方法 BtnClick()