Unity3D游戏开发最佳实践20技能(两)

简介:

【扩展和MonoBehaviourBase】
21、扩展一个自己的Mono Behaviour基类。然后自己的全部组件都从它派生
这能够使你方便的实现一些通用函数。比如类型安全的Invoke。或者是一些更复杂的调用(比如random等等)。
22、为Invoke, StartCoroutine and Instantiate 定义安全调用方法
定义一个托付任务(delegate Task),用它来定义须要调用的方法,而不要使用字符串属性方法名称。比如:
public void Invoke(Task task, float time){   Invoke(task.Method.Name, time);}
23、为共享接口的组件扩展
有些时候把获得组件、查找对象实如今一个组件的接口中会非常方便。


以下这样的实现方案使用了typeof。而不是泛型版本号的函数。

泛型函数无法在接口上工作,而typeof能够。以下这样的方法把泛型方法整洁的包装起来。

[C#] 
?
1
//Defined in the common base class for all mono behaviourspublic I GetInterfaceComponent () where I : class{  return GetComponent(typeof(I)) as I;} public static List FindObjectsOfInterface () where I : class{  MonoBehaviour[] monoBehaviours = FindObjectsOfType();  List list = new List ();   foreach(MonoBehaviour behaviour in monoBehaviours)  {     I component = behaviour.GetComponent(typeof(I)) as I;      if(component != null)     {        list.Add(component);     }  }   return list;}

24、使用扩展来让代码书写更便捷
比如:
[AppleScript] 
1
public staticclassCSTransform{  public static void SetX(this Transform transform,float x)  {     Vector3newPosition=         newVector3(x,transform.position.y,transform.position.z);      transform.position=newPosition;  }  ...}

25、使用防御性的GetComponent()
有些时候强制性组件依赖(通过RequiredComponent)会让人蛋疼。

比如。非常难在Inspector中改动组件(即使他们有相同的基类)。以下是一种替代方案。当一个必要的组件没有找到时。输出一条错误信息。


[AppleScript] 
?
1
public static T GetSafeComponent<</code>T>(this GameObject obj)whereT:MonoBehaviour{  T component=obj.GetComponent<</code>T>();   if(component==null)  {     Debug.LogError("Expected to find component of type "         +typeof(T)+" but found none",obj);  }   returncomponent;}
【风格】 26、避免对同一件事使用不同的处理风格
在非常多情况下,某件事并不仅仅有一个惯用手法。在这样的情况下,在项目中明白选择当中的一个来使用。以下是原因:
  • 一些做法并不能非常好的一起协作。

    使用一个。能强制统一设计方向。并明白指出不是其它做法所指的方向。

  • 团队成员使用统一的风格。可能方便大家互相的理解。

    他使得总体结构和代码都更easy理解。这也能够降低错误;

几组风格的样例:
  • 协程与状态机(Coroutines vs. state machines);
  • 嵌套的Prefab、互相链接的Prefab、超级Prefab(Nested prefabs vs. linked prefabs vs. God prefabs);
  • 数据分离的策略。
  • 在2D游戏的使用Sprite的方法;
  • Prefab的结构;
  • 对象生成策略;
  • 定位对象的方法:使用类型、名称、层、引用关系;
  • 对象分组的方法:使用类型、名称、层、引用数组。
  • 找到一组对象,还是让它们自己来注冊。
  • 控制运行次序(使用Unity的运行次序设置。还是使用Awake/Start/Update/LateUpdate,还是使用纯手动的方法,或者是次序无关的架构)。
  • 在游戏中使用鼠标选择对象/位置/目标:SelectionManager或者是对象自主管理;
  • 在场景变换时保存数据:通过PlayerPrefs,或者是在新场景载入时不要销毁的对象;
  • 组合动画的方法:混合、叠加、分层;


                                                                                                                                                     
【时间】
27、维护一个自己的Time类,能够使游戏暂停更easy实现
做一个“Time.DeltaTime”和""Time.TimeSinceLevelLoad"的包装,用来实现暂停和游戏速度缩放。这使用起来略显麻烦,可是当对象执行在不同的时钟速率下的时候就方便多了(比如界面动画和游戏内动画)。

                                                                                                                                                     
【生成对象】
28、不要让游戏执行时生成的对象搞乱场景层次结构
在游戏执行时,为动态生成的对象设置好它们的父对象。能够让你更方便的查找。

你能够使用一个空的对象,或者一个没有行为的单件来简化代码中的訪问。能够给这个对象命名为“DynamicObjects”。


                                                                                                                                                    
【类设计】
29、使用单件(Singleton)模式
从以下这个类派生的全部类,将自己主动获得单件功能:

[AppleScript] 
1
publicclassSingleton<</code>T>:MonoBehaviourwhereT:MonoBehaviour{  protected static T instance;   /**     Returnstheinstanceofthis singleton.  */  public static T Instance  {     get     {        if(instance==null)        {           instance=(T)FindObjectOfType(typeof(T));            if(instance==null)           {              Debug.LogError("An instance of "+typeof(T)+                  " is needed in the scene, but there is none.");           }        }         returninstance;     }  }}


单件能够作为一些管理器,比如ParticleManager或者AudioManager亦或者GUIManager。
  • 对于那些非唯一的prefab实例使用单件管理器(比如Player)。不要为了坚持这条原则把类的层次关系复杂化。宁愿在你的GameManager(或其它合适的管理器中)中持有一个它们的引用。
  • 对于外部常常使用的共同拥有变量和方法定义为static。这样你能够这样简便的书写“GameManager.Player”。而不用写成“GameManager.Instance.player”。

30、在组件中不要使用public成员变量,除非它须要在inspector中调节
除非须要设计师(策划or美术)去调节的变量。特别是它不能明白表明自己是做什么的变量。不要声明为public。假设在这些特殊情况下,无法避免,则可使用两个甚至四个下划线来表明不要从外部调节它,比如:
public float __aVariable; 
31、把界面和游戏逻辑分开
这一条本质上就是指的MVC模式。
全部的输入控制器,仅仅负责向对应的组件发送命令。让它们知道控制器被调用了。举一个控制器逻辑的样例,一个控制器依据玩家的状态来决定发送哪个命令。

可是这样并不好(比如,假设你加入了多个控制器。那将会导致逻辑反复)。相反的,玩家对象应该依据当前状态(比如减速、惊恐)来设置当前的速度。并依据当前的面朝向来计算怎样向前移动。控制器仅仅负责做他们自己状态相关的事情,控制器不改变玩家的状态。因此控制前甚至能够根本不知道玩家的状态。另外一个样例,切换武器。

正确的方法是,玩家有一个函数:“SwitchWeapon(Weapon newWeapon)”供GUI调用。GUI不应该维护全部对象的Transform和他们之间的父子关系。


全部界面相关的组件,仅仅负责维护和处理他们自己状态相关的数据。比如,显示一个地图,GUI能够依据玩家的位移计算地图的显示。可是。这是游戏状态数据。它不属于GUI。GUI仅仅是显示游戏状态数据。这些数据应该在其它地方维护。地图数据也应该在其它地方维护(比如GameManager)。


游戏玩法对象不应该关心GUI。有一个例外是处理游戏暂停(可能是通过控制Time.timeScale,事实上这并非个好主意)。游戏玩法对象应该知道游戏是否暂停。可是,这就是所有了。

另外。不要把GUI组件挂到游戏玩法对象上。


这么说吧。假设你把全部的GUI类都删了,游戏应该能够正确编译。


你还应该达到:在不须要重写游戏逻辑的前提下,重写GUI和输入控制。



32、分离状态控制和簿记变量
簿记变量仅仅是为了使用起来方便或者提高查找速度。而且能够依据状态控制来覆盖。将两者分离能够简化:

  • 保存游戏状态
  • 调试游戏状态
实现方法之中的一个是为每一个游戏逻辑定义一个”SaveData“类,比如:
[AppleScript] 
?
1
[Serializable]PlayerSaveData{  public float health;//publicforserialisation,notexposedininspector} Player{  //... bookkeeping variables   //Don’t exposestateininspector. Stateis nottweakable.  private PlayerSaveData playerSaveData;}

33、分离特殊的配置
如果我们有两个敌人,它们使用同一个Mesh。可是有不同的属性设置(比如不同的力量、不同的速度等等)。有非常多方法来分离数据。以下是我比較喜欢的一种。特别是对于对象生成或者游戏存档时,会非常好用。

(属性设置不是状态数据。而是配置数据。所以我们不须要存档他们。

当对象载入或者生成是,属性设置会自己主动载入。)

  • 为每个游戏逻辑类定义一个模板类。

    比如。对于敌人。我们来一个“EnemyTemplate”。全部的属性设置变量都保存在这个类中。

  • 在游戏逻辑的类中,定义一个上述模板类型的变量。
  • 制作一个敌人的Prefab。以及两个模板的Prefab:“WeakEnemyTemplate”和"StrongEnemyTemplate"。

  • 在载入或者生成对象是,把模板变量正确的复制。


这样的方法可能有点复杂(在一些情况下,可能不须要这样)。

举个样例,最好使用泛型。我们能够这样定义我们的类:
[AppleScript] 
?
1
publicclassBaseTemplate{  ...}publicclassActorTemplate:BaseTemplate{  ...}publicclassEntity<</code>EntityTemplateType>whereEntityTemplateType:BaseTemplate{  EntityTemplateType template;  ...}publicclassActor:Entity<</code>ActorTemplate>{  ...}

34、除了显示用的文本。不要使用字符串
特别是不要用字符串作为对象或者prefab等等的ID标识。一个非常遗憾的例外是动画系统,须要使用字符串来訪问对应的动画。
35、避免使用public的数组
举例说明,不要定义一个武器的数组,一个子弹的数组,一个粒子的数组,这样你的代码看起来像这样:
public void SelectWeapon(int index){    currentWeaponIndex = index;   Player.SwitchWeapon(weapons[currentWeapon]);} public void Shoot(){   Fire(bullets[currentWeapon]);  FireParticles(particles[currentWeapon]);   }
这在代码中还不是什么大问题。可是在Inspector中设置他们的值的时候,就非常难不犯错了。

我们能够定义一个类,来封装这三个变量,然后使用一个它的实例数组:
[Serializable]public class Weapon{   public GameObject prefab;   public ParticleSystem particles;   public Bullet bullet;} 这样代码看起来非常整洁。可是更重要的是,在Inspector中设置时就不easy犯错了。

36、在结构中避免使用数组
举个样例,一个玩家能够有三种攻击形式,每种使用当前的武器,并发射不同的子弹、产生不同的行为。
你能够把三个子弹作为一个数组。并像以下这样组织逻辑:
public void FireAttack(){   /// behaviour   Fire(bullets[0]);} public void IceAttack(){   /// behaviour   Fire(bullets[1]);} public void WindAttack(){   /// behaviour   Fire(bullets[2]);} 使用枚举值能够让代码看起来更好一点:

public void WindAttack(){   /// behaviour   Fire(bullets[WeaponType.Wind]);} 
可是这对Inspector一点也不好。

最好使用单独的变量,而且起一个好的变量名,可以代表他们的内容的含义。使用以下这个类会更整洁。


[Serializable]public class Bullets{   public Bullet FireBullet;   public Bullet IceBullet;   public Bullet WindBullet;}这里如果没有其它的Fire、Ice、Wind的数据。

37、把数据组织到可序列化的类中,能够让inspector更整洁
有些对象有一大堆可调节的变量,这样的情况下在Inspector中找到某个变量简直就成了噩梦。

为了简化这样的情况,能够使用一下的步骤:

  • 把这些变量分组定义到不同的类中。并让它们声明为public和serializable。
  • 在一个基本的类中,把上述类的实例定义为public成员变量;
  • 不用在Awake或者Start中初始化这些变量,由于Unity会处理好它们。
  • 你能够定义它们的默认值;
这能够把变量分组到Inspector的分组页签中,方便管理。

[Serializable]public class MovementProperties //Not a MonoBehaviour!{   public float movementSpeed;   public float turnSpeed = 1; //default provided} public class HealthProperties //Not a MonoBehaviour!{   public float maxHealth;   public float regenerationRate;} public class Player : MonoBehaviour{   public MovementProperties movementProeprties;   public HealthPorperties healthProeprties;}


版权声明:本文博客原创文章,博客,未经同意,不得转载。






本文转自mfrbuaa博客园博客,原文链接http://www.cnblogs.com/mfrbuaa/p/4633299.html,如需转载请自行联系原作者


相关文章
|
2月前
|
测试技术 C# 图形学
掌握Unity调试与测试的终极指南:从内置调试工具到自动化测试框架,全方位保障游戏品质不踩坑,打造流畅游戏体验的必备技能大揭秘!
【9月更文挑战第1天】在开发游戏时,Unity 引擎让创意变为现实。但软件开发中难免遇到 Bug,若不解决,将严重影响用户体验。调试与测试成为确保游戏质量的最后一道防线。本文介绍如何利用 Unity 的调试工具高效排查问题,并通过 Profiler 分析性能瓶颈。此外,Unity Test Framework 支持自动化测试,提高开发效率。结合单元测试与集成测试,确保游戏逻辑正确无误。对于在线游戏,还需进行压力测试以验证服务器稳定性。总之,调试与测试贯穿游戏开发全流程,确保最终作品既好玩又稳定。
93 4
|
2月前
|
设计模式 存储 人工智能
深度解析Unity游戏开发:从零构建可扩展与可维护的游戏架构,让你的游戏项目在模块化设计、脚本对象运用及状态模式处理中焕发新生,实现高效迭代与团队协作的完美平衡之路
【9月更文挑战第1天】游戏开发中的架构设计是项目成功的关键。良好的架构能提升开发效率并确保项目的长期可维护性和可扩展性。在使用Unity引擎时,合理的架构尤为重要。本文探讨了如何在Unity中实现可扩展且易维护的游戏架构,包括模块化设计、使用脚本对象管理数据、应用设计模式(如状态模式)及采用MVC/MVVM架构模式。通过这些方法,可以显著提高开发效率和游戏质量。例如,模块化设计将游戏拆分为独立模块。
170 3
|
2月前
|
图形学 开发工具 git
Unity与版本控制:游戏开发团队如何利用Git打造高效协作流程,实现代码管理的最佳实践指南
【8月更文挑战第31天】版本控制在软件开发中至关重要,尤其在Unity游戏开发中,能提升团队协作效率并避免错误。本文介绍如何在Unity项目中应用版本控制的最佳实践,包括选择Git、配置项目以排除不必要的文件、组织项目结构、避免冲突、规范提交信息以及使用分支管理开发流程,从而提高代码质量和团队协作效率。
182 1
|
3月前
|
图形学 机器学习/深度学习 人工智能
颠覆传统游戏开发,解锁未来娱乐新纪元:深度解析如何运用Unity引擎结合机器学习技术,打造具备自我进化能力的智能游戏角色,彻底改变你的游戏体验——从基础设置到高级应用全面指南
【8月更文挑战第31天】本文探讨了如何在Unity中利用机器学习增强游戏智能。作为领先的游戏开发引擎,Unity通过ML-Agents Toolkit等工具支持AI代理的强化学习训练,使游戏角色能自主学习完成任务。文章提供了一个迷宫游戏示例及其C#脚本,展示了环境观察、动作响应及奖励机制的设计,并介绍了如何设置训练流程。此外,还提到了Unity与其他机器学习框架(如TensorFlow和PyTorch)的集成,以实现更复杂的游戏玩法。通过这些技术,游戏的智能化程度得以显著提升,为玩家带来更丰富的体验。
61 1
|
2月前
|
图形学 开发者
透视与正交之外的奇妙视界:深入解析Unity游戏开发中的相机与视角控制艺术,探索打造沉浸式玩家体验的奥秘与技巧
【8月更文挑战第31天】在Unity中,相机不仅是玩家观察游戏世界的窗口,更是塑造氛围和引导注意力的关键工具。通过灵活运用相机系统,开发者能大幅提升游戏的艺术表现力和沉浸感。本文将探讨如何实现多种相机控制,包括第三人称跟随和第一人称视角,并提供实用代码示例。
126 0
|
2月前
|
图形学 开发者
【独家揭秘】Unity游戏开发秘籍:从基础到进阶,掌握材质与纹理的艺术,打造超现实游戏视效的全过程剖析——案例教你如何让每一面墙都会“说话”
【8月更文挑战第31天】Unity 是全球领先的跨平台游戏开发引擎,以其高效性能和丰富的工具集著称,尤其在提升游戏视觉效果方面表现突出。本文通过具体案例分析,介绍如何利用 Unity 中的材质与纹理技术打造逼真且具艺术感的游戏世界。材质定义物体表面属性,如颜色、光滑度等;纹理则用于模拟真实细节。结合使用两者可显著增强场景真实感。以 FPS 游戏为例,通过调整材质参数和编写脚本动态改变属性,可实现自然视觉效果。此外,Unity 还提供了多种高级技术和优化方法供开发者探索。
50 0
|
2月前
|
图形学 开发者 UED
Unity游戏开发必备技巧:深度解析事件系统运用之道,从生命周期回调到自定义事件,打造高效逻辑与流畅交互的全方位指南
【8月更文挑战第31天】在游戏开发中,事件系统是连接游戏逻辑与用户交互的关键。Unity提供了多种机制处理事件,如MonoBehaviour生命周期回调、事件系统组件及自定义事件。本文介绍如何有效利用这些机制,包括创建自定义事件和使用Unity内置事件系统提升游戏体验。通过合理安排代码执行时机,如在Awake、Start等方法中初始化组件,以及使用委托和事件处理复杂逻辑,可以使游戏更加高效且逻辑清晰。掌握这些技巧有助于开发者更好地应对游戏开发挑战。
112 0
|
2月前
|
图形学 开发者 搜索推荐
Unity Asset Store资源大解密:自制与现成素材的优劣对比分析,教你如何巧用海量资产加速游戏开发进度
【8月更文挑战第31天】游戏开发充满挑战,尤其对独立开发者或小团队而言。Unity Asset Store 提供了丰富的资源库,涵盖美术、模板、音频和脚本等,能显著加快开发进度。自制资源虽具个性化,但耗时长且需专业技能;而 Asset Store 的资源经官方审核,质量可靠,可大幅缩短开发周期,使开发者更专注于核心玩法。然而,使用第三方资源需注意版权问题,且可能需调整以适应特定需求。总体而言,合理利用 Asset Store 能显著提升开发效率和项目质量。
63 0
|
3月前
|
开发者 图形学 API
从零起步,深度揭秘:运用Unity引擎及网络编程技术,一步步搭建属于你的实时多人在线对战游戏平台——详尽指南与实战代码解析,带你轻松掌握网络化游戏开发的核心要领与最佳实践路径
【8月更文挑战第31天】构建实时多人对战平台是技术与创意的结合。本文使用成熟的Unity游戏开发引擎,从零开始指导读者搭建简单的实时对战平台。内容涵盖网络架构设计、Unity网络API应用及客户端与服务器通信。首先,创建新项目并选择适合多人游戏的模板,使用推荐的网络传输层。接着,定义基本玩法,如2D多人射击游戏,创建角色预制件并添加Rigidbody2D组件。然后,引入网络身份组件以同步对象状态。通过示例代码展示玩家控制逻辑,包括移动和发射子弹功能。最后,设置服务器端逻辑,处理客户端连接和断开。本文帮助读者掌握构建Unity多人对战平台的核心知识,为进一步开发打下基础。
113 0
|
3月前
|
开发者 图形学 C#
深度解密:Unity游戏开发中的动画艺术——Mecanim状态机如何让游戏角色栩栩如生:从基础设置到高级状态切换的全面指南,助你打造流畅自然的游戏动画体验
【8月更文挑战第31天】Unity动画系统是游戏开发的关键部分,尤其适用于复杂角色动画。本文通过具体案例讲解Mecanim动画状态机的使用方法及原理。我们创建一个游戏角色并设计行走、奔跑和攻击动画,详细介绍动画状态机设置及脚本控制。首先导入动画资源并添加Animator组件,然后创建Animator Controller并设置状态间的转换条件。通过编写C#脚本(如PlayerMovement)控制动画状态切换,实现基于玩家输入的动画过渡。此方法不仅适用于游戏角色,还可用于任何需动态动画响应的对象,增强游戏的真实感与互动性。
88 0
下一篇
无影云桌面