博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Unity 游戏框架搭建 (七) 减少加班利器-QApp类
阅读量:6036 次
发布时间:2019-06-20

本文共 4657 字,大约阅读时间需要 15 分钟。

本来这周想介绍一些框架中自认为比较好用的小工具的,但是发现很多小工具都依赖一个类----App。

App 类的职责:

1.接收 Unity 的生命周期事件。

2.做为游戏的入口。

3.一些框架级别的组件初始化。

本文只介绍App的职责2:做为游戏的入口。

Why?

在我小时候做项目的时候,每次改一点点代码(或者不止一点点),要看下结果就要启动游戏->Loading界面->点击各种按钮->跳转到目标界面看结果或者Log之类的。一天如果10次这种行为会浪费很多时间,如果按照时薪算的话那就是......很多钱(捂嘴)。 流程图是这样的:

为什么会出现这种问题呢?

1.模块间的耦合度太高了。下一个模块要依赖前一个模块的一些数据或者逻辑。 2.或者有可能是这个模块设计得太大了,界面太多,也会发生这种情况。

解决方案:

针对问题1:在模块的入口提供一个测试的接口,用来写这个模块的资源加载或者数据初始化的逻辑,...什么!?...你们项目就一个模块...来来来我们好好聊聊..... 针对问题2:在模块的入口提供一个测试接口,写跳转到目标界面的相关代码。 流程图是这样的:

虽然很low但是勉强解决了问题。

阶段的划分

资源加载乱七八糟的代码和最好能一步跳转到目标界面的代码,需要在出包或者跑完整游戏流程的时候失效。 如何做到?答案是阶段的划分。 我的框架里分为如下几个阶段: 1.开发阶段: 不断的编码->验证结果->编码->验证结果->blablabla。 2.出包/真机阶段: 这个阶段跑跑完整流程,在真机上跑跑,给QA测测。 3.发布阶段: 上线了,yeah!。

对应的枚举:

public enum AppMode {	Developing,	QA,	Release}复制代码

很明显,乱七八糟的代码是要在开发阶段有效,但是在QA或者Release版本中无效的。那么只要在游戏的入口处判断当前在什么阶段就好了。 开始编码:

/// /// 全局唯一继承于MonoBehaviour的单例类,保证其他公共模块都以App的生命周期为准/// public class App : QMonoSingleton
{ public AppMode mode = AppMode.Developing; private App() {} void Awake() { // 确保不被销毁 DontDestroyOnLoad(gameObject); mInstance = this; // 进入欢迎界面 Application.targetFrameRate = 60; } void Start() { CoroutineMgr.Instance ().StartCoroutine (ApplicationDidFinishLaunching()); } ///
/// 进入游戏 /// IEnumerator ApplicationDidFinishLaunching() { // 配置文件加载 类似PlayerPrefs QSetting.Load(); // 日志输出 QLog.Instance (); yield return GameManager.Instance ().Init (); // 进入测试逻辑 if (App.Instance().mode == AppMode.Developing) { // 测试资源加载 ResMgr.Instance ().LoadRes ("TestRes",delegate(string resName, Object resObj) { if (null != resObj) { GameObject.Instantiate(resObj); } // 进入目标界面等等 }); yield return null; // 进入正常游戏逻辑 } else { yield return GameManager.Instance ().Launch (); } yield return null; }复制代码

首先App是Mono单例,要接收Unity的生命周期. 然后要维护一个AppMode类型的变量,便于区分。 之后在,ApplicationDidFinishLaunching中有这么一段代码

// 进入测试逻辑		if (App.Instance().mode == AppMode.Developing) 		{			// 测试资源加载			ResMgr.Instance ().LoadRes ("TestRes",delegate(string resName, Object resObj) 			{				if (null != resObj) 				{					GameObject.Instantiate(resObj);				}				// 进入目标界面等等			});			yield return null;		// 进入正常游戏逻辑		} 		else 		{			yield return GameManager.Instance ().Launch ();		}复制代码

在这段代码中做了阶段的区分。所有的逻辑都可以写在这里。这样基本的需求就满足啦。

####还有一个问题: 假如一个游戏的业务逻辑分为模块A,B,C,D,E,分为5个不同的人来开发,那App是一个mono单例,除非不提交App代码,否则每次都要解决冲突,同样很浪费时间。怎么办? 答案是通过多态来解决,先定义一个ITestEntry接口,只定义一个方法。

/// 	/// 测试入口	/// 	public interface ITestEntry 	{		/// 		/// 启动		/// 		IEnumerator Launch();	}复制代码

然后每个模块分别实现ITestEntry接口,例如AModuleTestEntry,BModuleTestEntry等等。 看下项目中的实现:

/// /// AR模块测试入口/// public class ARSceneTestEntry :MonoBehaviour,ITestEntry {	public IEnumerator Launch() 	{		Debug.LogWarning ("进入AR场景开始");		yield return GameObject.Find ("ARScene").GetComponent
().Launch (); yield return null; }}复制代码

App类中阶段区分的代码要改成这样:

// 进入测试逻辑		if (App.Instance().mode == AppMode.Developing) {					yield return GetComponent
().Launch (); // 进入正常游戏逻辑 } else { yield return GameManager.Instance ().Launch (); }复制代码

因为Launch方法的返回类型是IEnumerator,所以很好控制跳转的时间。 看下在Unity中是什么样的:

每个模块都要有个App的GameObject,原因是因为,框架的其他的组件依赖于App,也想过把依赖的部分抽离出来,那样的话可能命名为QMonoLifeCircleReceiver和ModuleEntry之类的,这样遵循了单一职责原则。不过孰优孰略各有千秋。我觉得叫App更直观一些,因为入口、组件初始化、启动某个模块应该是通常放在一起更人性化,还有一些ApplicationDidEnterBackground之类的事件还是模仿iOS的AppDelegate人性化一些。

如果要跑完整流程,那么把模块的App GameObject关掉就好了。要注意一点是:在整个游戏的入口场景要有个App GameObject放在上面,并且AppMode要为Release或者QA。这样才能正常地跑起来。

OK就这样....

对未来的一些畅想:

  1. 最近在想着如何为项目引入自动化测试,有一个思路是这样的,界面的所有输入包括点击事件等都包装成一个命令或者一个消息。测试的时候只要不断地自动发送消息或者命令就好了。当然只是个畅想。 那和这个有毛关系呢,有啊!界面跳转的时候可以发命令或者消息就够了啊,这样还很方便。 但实际上有很多问题,包括模块的最上层如何拿到一些界面组件的权限比如按钮等等。处理命令或者消息的话那么所有的输入都要经过一层过滤。。。。额。。想想好麻烦。。。以后吧。。。以后吧。。

  2. 框架的很多组件都是基于字典实现的。字典真好用,23333。以后还是想办法能改的都改成List吧。

欢迎讨论!

相关链接:

:https://github.com/liangxiegame/QFramework

:https://github.com/liangxiegame/QFramework/tree/master/Assets/HowToWriteUnityGameFramework/

QFramework&游戏框架搭建QQ交流群: 623597263

转载请注明地址:http://liangxiegame.com/

微信公众号:liangxiegame

如果有帮助到您:

如果觉得本篇教程或者 QFramework 对您有帮助,不妨通过以下方式赞助笔者一下,鼓励笔者继续写出更多高质量的教程,也让更多的力量加入 QFramework 。

  • 给 QFramework 一个 Star:https://github.com/liangxiegame/QFramework
  • 下载 Asset Store 上的 QFramework 给个五星(如果有评论小的真是感激不尽):http://u3d.as/SJ9
  • 购买 gitchat 话题并给 5 星好评: http://gitbook.cn/gitchat/activity/5abc3f43bad4f418fb78ab77 (6 元,会员免费)
  • 购买同名的蛮牛视频课程并给 5 星好评:http://edu.manew.com/course/431 (目前定价 29.8 元)
  • 购买同名电子书 :https://www.kancloud.cn/liangxiegame/unity_framework_design( 29.9 元,内容会在 2018 年 10 月份完结)

笔者在这里保证 QFramework、入门教程、文档和此框架搭建系列的专栏永远免费开源。以上捐助产品的内容对于使用 QFramework 的使用来讲都不是必须的,所以大家不用担心,各位使用 QFramework 或者 阅读此专栏 已经是对笔者团队最大的支持了。

你可能感兴趣的文章
个人关于vue全家桶开发规范的梳理
查看>>
【论文实现】一篇Sigkdd的弹幕分析论文的python实现【LDA 实践者】
查看>>
十几行代码教你实现一个最简版的promise
查看>>
Java IO学习笔记八
查看>>
systemd vs supervisord
查看>>
如何构建一个分布式爬虫:基础篇
查看>>
50 行代码的 HTML 编译器
查看>>
[GitHub] vim 实操教程
查看>>
常见网络攻击--XSS && CSRF
查看>>
在项目中使用echarts
查看>>
Akka系列(四):Akka中的共享内存模型
查看>>
腾讯云语音合成TTS
查看>>
[LeetCode] Find Largest Value in Each Tree Row
查看>>
【Sublime Text】安装插件
查看>>
javascript正则表达式
查看>>
快速初始化express项目
查看>>
EGO走进美团——追寻千亿市场背后的技术力量
查看>>
将敏捷应用于工业机械开发
查看>>
中台之上(六):如何为一个商业银行设计业务架构?
查看>>
开源项目koa-router被叫卖,周下载10W+只要5000美元
查看>>