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

转载请注明地址:凉鞋的笔记

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

App类的职责:

1.接收Unity的生命周期事件。
  2.做为游戏的入口。
  3.一些框架级别的组件初始化。

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

Why?

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

01.png

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

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

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

02.png

虽然很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等等。
看下项目中的实现:

/// <summary>
/// AR模块测试入口
/// </summary>
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中是什么样的:

03.png

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

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

OK就这样....

对未来的一些畅想:

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

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 195,980评论 5 462
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 82,422评论 2 373
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 143,130评论 0 325
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,553评论 1 267
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,408评论 5 358
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,326评论 1 273
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,720评论 3 386
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,373评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,678评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,722评论 2 312
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,486评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,335评论 3 313
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,738评论 3 299
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,009评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,283评论 1 251
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,692评论 2 342
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,893评论 2 335

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,132评论 25 707
  • 发现 关注 消息 iOS 第三方库、插件、知名博客总结 作者大灰狼的小绵羊哥哥关注 2017.06.26 09:4...
    肇东周阅读 11,952评论 4 60
  • 车上听精英日课,听到一个词 非常有趣: “注意力商人”,说的是现在无论是广告还是媒体,都在想办法占领我们的注意力,...
    逄格亮阅读 274评论 0 1
  • 真正的自由不是外在的,而是内在的。我觉得人生模式就像绑在我们身上的绳子一样,让我们动弹不得,而且我们像一个傀儡娃娃...
    园苗的春天阅读 155评论 0 1