本来这周想介绍一些框架中自认为比较好用的小工具的,但是发现很多小工具都依赖一个类----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版本中无效的。那么只要在游戏的入口处判断当前在什么阶段就好了。
开始编码:

/// <summary>
/// 全局唯一继承于MonoBehaviour的单例类,保证其他公共模块都以App的生命周期为准
/// </summary>
public class App : QMonoSingleton<App>
{public AppMode mode = AppMode.Developing;private App() {}void Awake(){// 确保不被销毁
        DontDestroyOnLoad(gameObject);mInstance = this;// 进入欢迎界面Application.targetFrameRate = 60;}void  Start(){CoroutineMgr.Instance ().StartCoroutine (ApplicationDidFinishLaunching());}/// <summary>/// 进入游戏/// </summary>
    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接口,只定义一个方法。
/// <summary>
/// 测试入口
/// </summary>
public interface ITestEntry
{/// <summary>/// 启动/// </summary>
    IEnumerator Launch();
}

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

///
/// AR模块测试入口
///
public class ARSceneTestEntry :MonoBehaviour,ITestEntry
{
public IEnumerator Launch()
{

    Debug.LogWarning ("进入AR场景开始");yield return GameObject.Find ("ARScene").GetComponent<ARScene> ().Launch ();yield return null;
}

}


App类中阶段区分的代码要改成这样:
    // 进入测试逻辑if (App.Instance().mode == AppMode.Developing) {yield return GetComponent<ITestEntry> ().Launch ();// 进入正常游戏逻辑} else {yield return GameManager.Instance ().Launch ();}

因为Launch方法的返回类型是IEnumerator,所以很好控制跳转的时间。
看下在Unity中是什么样的:
![](/content/images/2016/07/-----2016-07-03---8-37-42.png)每个模块都要有个App的GameObject,原因是因为,框架的其他的组件依赖于App,也想过把依赖的部分抽离出来,那样的话可能命名为QMonoLifeCircleReceiver和ModuleEntry之类的,这样遵循了单一职责原则。不过孰优孰略各有千秋。我觉得叫App更直观一些,因为入口、组件初始化、启动某个模块应该是通常放在一起更人性化,还有一些ApplicationDidEnterBackground之类的事件还是模仿iOS的AppDelegate人性化一些。如果要跑完整流程,那么把模块的App GameObject关掉就好了。要注意一点是:在整个游戏的入口场景要有个App GameObject放在上面,并且AppMode要为Release或者QA。这样才能正常地跑起来。OK就这样....
#### 对未来的一些畅想:1. 最近在想着如何为项目引入自动化测试,有一个思路是这样的,界面的所有输入包括点击事件等都包装成一个命令或者一个消息。测试的时候只要不断地自动发送消息或者命令就好了。当然只是个畅想。 那和这个有毛关系呢,有啊!界面跳转的时候可以发命令或者消息就够了啊,这样还很方便。 但实际上有很多问题,包括模块的最上层如何拿到一些界面组件的权限比如按钮等等。处理命令或者消息的话那么所有的输入都要经过一层过滤。。。。额。。想想好麻烦。。。以后吧。。。以后吧。。2. 框架的很多组件都是基于字典实现的。字典真好用,23333。以后还是想办法能改的都改成List吧。#### 欢迎讨论!附:[我的框架QFramework地址](https://github.com/liangxiegame/QFramework)转载请注明地址:[凉鞋的笔记](http://liangxiegame.com/)微信公众号:liangxiegame![](http://liangxiegame.com/content/images/2017/06/qrcode_for_gh_32f0f3669ac8_430.jpg)#output/writing/Unity游戏框架搭建

转载于:https://www.cnblogs.com/liangxiegame/p/Unity-you-xi-kuang-jia-da-jian-qi-jian-shao-jia-ba.html

Unity 游戏框架搭建 (七) 减少加班利器-QApp类相关推荐

  1. Unity 游戏框架搭建 (九) 减少加班利器-QConsole

    为毛要实现这个工具? 在我小时候,每当游戏在真机运行时,我们看到的日志是这样的. 没高亮啊,还有乱七八糟的堆栈信息,好干扰日志查看,好影响心情. 还有就是必须始终连着usb线啊,我想要想躺着测试... ...

  2. Unity 游戏框架搭建 (二十一) 使用对象池时的一些细节

    上篇文章使用SafeObjectPool实现了一个简单的Msg类.代码如下: class Msg : IPoolAble,IPoolType{#region IPoolAble 实现public vo ...

  3. Unity 游戏框架搭建 (五) 简易消息机制

    什么是消息机制? 23333333,让我先笑一会. 为什么用消息机制? 三个字,解!!!!耦!!!!合!!!!. 我的框架中的消息机制用例: 1.接收者 using UnityEngine;names ...

  4. Unity 游戏框架搭建 2018 (四) 我所理解的框架

    前言 架构和框架这些概念听起来很遥远,让很多初学者不明觉厉.会产生"等自己技术牛逼了再去做架构或者搭建框架"这样的想法.在这里笔者可以很肯定地告诉大家,初学者是完全可以去做这些事情 ...

  5. Unity 游戏框架搭建 2018 (一) 架构、框架与 QFramework 简介

    约定 还记得上版本的第二十四篇的约定嘛?现在出来履行啦~ 为什么要重制? 之前写的专栏都是按照心情写的,在最初的时候笔者什么都不懂,而且文章的发布是按照很随性的一个顺序.结果就是说,大家都看完了,都还 ...

  6. Unity 游戏框架搭建 2019 (四十五) 独立的方法和独立的类

    我们在开始本示例之前,先整理出我们当前库中的代码类型. 工具方法:CommonUtil.GameObjectSimplify等. 类: MonoBehaviourSimplify. 静态方法中的方法全 ...

  7. Unity 游戏框架搭建 2019 (四十二) MonoBehaviour 简化

    在前两篇,我们完成了第九个示例.为了完善第九个示例,我们复习了类的继承,又学习了泛型和 params 关键字. 我们已经接触了类的继承了.接触继承之前,把类仅仅当做是方法的集合,接触了继承之后,我们的 ...

  8. Unity 游戏框架搭建 2019 (四十七) 集成到 MonoBehaviourSimplify

    还记得我们的简易消息机制是为了解决什么问题诞生的嘛? 是为了解决脚本间访问的问题. 我们回过头再看下 A 脚本如果想访问 B 脚本,使用消息机制,如何实现. 代码如下: public class A ...

  9. unity游戏框架学习-框架结构

    转眼毕业三年了,算上实习差不多四年的游戏开发了,一直想自己鼓捣套框架,奈何能力太次,不知道从哪开始.但是万事开头难,总要踏出第一步,才会有后面的两步,三步- 我认为的unity游戏框架就是一整套的工具 ...

最新文章

  1. Linux文件基本属性
  2. 湘潭大学网络编程_湘潭大学计算机学院网络空间安全学院“湘韵”研究生论坛成功举行...
  3. Java编程——服务器设计方案之应用限流
  4. laravel的路由分组,中间件,命名空间,子域名,路由前缀(四)
  5. Linux nc命令详解
  6. php mysql中华图书购物商城
  7. 多级联动下拉java,下拉列表多级联动前端实现
  8. NLP --- 隐马尔可夫HMM(概念详解、三个基本问题详解)
  9. idea拉出Output窗口和还原窗口
  10. 推荐系统专利:一种信息推荐系统及方法
  11. 数据库生成数据字典工具(PDMREAD)图解
  12. triton部署yolov5笔记(四)
  13. 如何成为一个合格的数据分析师
  14. JDK7和JDK8的区别
  15. snipaste滚动截图方法_Snipaste——这是我用过最好用的截图贴图工具!
  16. clickhouse分布式集群遇到的一些错误处理
  17. 图片文件探究1:各种图片文件十六进制的文件头和文件尾
  18. 凯翔:可以同时替代Nimble和Nutanix的存储
  19. <网络概述>——《计算机网络》
  20. Win10哪些服务可以关闭/禁用?

热门文章

  1. 秘密武器 | 看AnalyticDB如何强力支撑双十一
  2. sqlserver日期dateadd及STUFF等应用
  3. 创意总监分享:我是如何做一款手游地图的
  4. 如何在手游中运用高逼格光影效果?
  5. 零基础学Python(第十二章 列表【list】)
  6. 美团在Redis上踩过的一些坑-3.redis内存占用飙升
  7. sqlplus中利用spool生成带日期的文件名
  8. Django组件-中间件
  9. 数据库 Linux下的MySQL数据库管理
  10. 19、Java并发编程:线程间协作的两种方式:wait、notify、notifyAll和Condition