版权申明:

  • 本文原创首发于以下网站:
  1. 博客园『优梦创客』的空间:https://www.cnblogs.com/raymondking123
  2. 优梦创客的官方博客:https://91make.top
  3. 优梦创客的游戏讲堂:https://91make.ke.qq.com
  4. 『优梦创客』的微信公众号:umaketop
  • 您可以自由转载,但必须加入完整的版权声明!

万物起源:Init.cs

  • 打开范例场景init.unity,可以发现其场景层级如下:

    • 其中唯一重要的就是Global对象上挂在的init.cs脚本,关于其基础代码分析,还是建议大家看初见的教程(ghithub有链接)
    • 在这里只想重点分析大家一定会关心的一个问题:init.cs是如何加载初始界面的

init.cs是如何加载初始界面的:

  • 上节课分析了,init.cs首先加载UILoading界面,其加载流程大致是这样的,先上序列图,稍后结合序列图贴代码分析:
sequenceDiagram
Unity->> +Init: StartAsync
Init ->> BundleHelper: DownloadBundle()
BundleHelper->>EventSystem: Run(EventIdType.LoadingBegin)
EventSystem->>LoadingBeginEvent_CreateLoadingUI: Run()
LoadingBeginEvent_CreateLoadingUI->>UILoadingFactory: Create()
note right of UILoadingFactory: 实例化UILoading预制体,并附加UILoadingComponent(更新并显示加载进度)
Init->>-Unity: StartAsync
  • 加载初始界面的几个步骤如下:
  1. 调用EventSystem.Run(EventIdType.LoadingBegin)引发LoadingBegin事件:
public static class BundleHelper
{
public static async ETTask DownloadBundle()
{
if (Define.IsAsync)
{
try
{
using (BundleDownloaderComponent bundleDownloaderComponent = Game.Scene.AddComponent<BundleDownloaderComponent>())
{
await bundleDownloaderComponent.StartAsync(); Debug.Log("EventIdType.LoadingBegin");
Game.EventSystem.Run(EventIdType.LoadingBegin); await bundleDownloaderComponent.DownloadAsync();
} Game.EventSystem.Run(EventIdType.LoadingFinish); Game.Scene.GetComponent<ResourcesComponent>().LoadOneBundle("StreamingAssets");
ResourcesComponent.AssetBundleManifestObject = (AssetBundleManifest)Game.Scene.GetComponent<ResourcesComponent>().GetAsset("StreamingAssets", "AssetBundleManifest");
}
catch (Exception e)
{
Log.Error(e);
} }
}
}
  • 由于在unity编辑器环境下IsAsync标志被设为false(在VS环境下选中IsAsync成员,右键→速览定义可见),也即异步加载资源才可见loading画面,所以实际上不会看到loading画面!
  • 第19行为等待异步加载完毕后引发LoadingFinish事件,其流程与LoadingBegin类似,请同学们自行分析!
  1. 实现LoadingBegin事件处理程序:

    [Event(EventIdType.LoadingBegin)]
    public class LoadingBeginEvent_CreateLoadingUI : AEvent
    {
    public override void Run()
    {
    UI ui = UILoadingFactory.Create();
    Game.Scene.GetComponent<UIComponent>().Add(ui);
    }
    }

    在这里有需要注意学习定义事件类的方法:

    1. 为一个类添加Event标志(参数填具体事件类型)

    2. 从AEvent继承

    3. 此时,ET就会自动将该类识别为一个事件处理类(通过反射机制),并在EventSystem.Run被调用时执行LoadingBeginEvent_CreateLoadingUI事件类的Run方法!

  2. 第六行代码UILoadingFactory.Create()负责创建UILoading界面,下面代码加了注释:

    public static class UILoadingFactory
    {
    public static UI Create()
    {
    try
    {
    // KV是Resources文件夹下存储的本地预制体资源,主要存储一些键值对数据
    // 从KV加载UIType.UILoading预制体,并实例化UI对象:
    GameObject bundleGameObject = ((GameObject)ResourcesHelper.Load("KV")).Get<GameObject>(UIType.UILoading);
    GameObject go = UnityEngine.Object.Instantiate(bundleGameObject);
    go.layer = LayerMask.NameToLayer(LayerNames.UI); // 创建UI这个Entity,并将上面创建的UI对象作为该Entity的图形表示
    UI ui = ComponentFactory.Create<UI, string, GameObject>(UIType.UILoading, go, false); // 添加UILoadingComponent,该组件负责更新loading进度并刷新显示
    ui.AddComponent<UILoadingComponent>();
    return ui;
    }
    catch (Exception e)
    {
    Log.Error(e);
    return null;
    }
    }
    }

    说明:

    - UI类是一个Entity类,Entity间接从Component类继承,但只有Entity类可以附加组件,Component类不行

    - Entity和Component的关系实际就是设计模式中的Composite模式

    - UI类可以复用,当你要创建一个UI时,在ET框架下只要:

    - 添加一个static的UI工厂类,并在其中定义一个static的Create方法,具体实现参照UILoadingFactory

    - 为该工厂添加一个新的UI组件(从Component类继承),并实现该组件的事件系统(见下文)

  3. 实现UILoadingComponent并实现该组件的事件系统:

    • UILoading组件
    public class UILoadingComponent : Component
    {
    public Text text;
    }
    • UILoading事件系统:
    [ObjectSystem]
    public class UiLoadingComponentAwakeSystem : AwakeSystem<UILoadingComponent>
    {
    public override void Awake(UILoadingComponent self)
    {
    self.text = self.GetParent<UI>().GameObject.Get<GameObject>("Text").GetComponent<Text>();
    }
    } [ObjectSystem]
    public class UiLoadingComponentStartSystem : StartSystem<UILoadingComponent>
    {
    public override void Start(UILoadingComponent self)
    {
    StartAsync(self).Coroutine();
    } public async ETVoid StartAsync(UILoadingComponent self)
    {
    TimerComponent timerComponent = Game.Scene.GetComponent<TimerComponent>();
    long instanceId = self.InstanceId;
    while (true)
    {
    await timerComponent.WaitAsync(1000); if (self.InstanceId != instanceId)
    {
    return;
    } BundleDownloaderComponent bundleDownloaderComponent = Game.Scene.GetComponent<BundleDownloaderComponent>();
    if (bundleDownloaderComponent == null)
    {
    continue;
    }
    self.text.text = $"{bundleDownloaderComponent.Progress}%";
    }
    }
    }

    事件类的定义:

    1. 添加[ObjectSystem]标志

    2. 继承自对应的XxxSystem类,并实现基类的虚方法

    - 事件类与Unity中含义类似,请自行参阅源码学习

总结:

  • 通过对UILoading的学习,我们已经接触了ET的一个完整的ECS对象:

    • E:Entity,对应UI类
    • C:Component,对应UILoadingComponent类
    • S:System,

      对应UiLoadingComponentAwakeSystem和

      UiLoadingComponentStartSystem类

Unity进阶之ET网络游戏开发框架 02-ET的客户端启动流程分析的更多相关文章

  1. Unity进阶之ET网络游戏开发框架 01-下载、运行

    版权申明: 本文原创首发于以下网站: 博客园『优梦创客』的空间:https://www.cnblogs.com/raymondking123 优梦创客的官方博客:https://91make.top ...

  2. Unity进阶之ET网络游戏开发框架 04-资源打包

    版权申明: 本文原创首发于以下网站: 博客园『优梦创客』的空间:https://www.cnblogs.com/raymondking123 优梦创客的官方博客:https://91make.top ...

  3. Unity进阶之ET网络游戏开发框架 05-搭建自己的第一个Scene

    版权申明: 本文原创首发于以下网站: 博客园『优梦创客』的空间:https://www.cnblogs.com/raymondking123 优梦创客的官方博客:https://91make.top ...

  4. Unity进阶之ET网络游戏开发框架 08-深入登录成功消息

    版权申明: 本文原创首发于以下网站: 博客园『优梦创客』的空间:https://www.cnblogs.com/raymondking123 优梦创客的官方博客:https://91make.top ...

  5. Unity进阶之ET网络游戏开发框架 03-Hotfix层启动

    版权申明: 本文原创首发于以下网站: 博客园『优梦创客』的空间:https://www.cnblogs.com/raymondking123 优梦创客的官方博客:https://91make.top ...

  6. Unity进阶之ET网络游戏开发框架 06-游客登录

    版权申明: 本文原创首发于以下网站: 博客园『优梦创客』的空间:https://www.cnblogs.com/raymondking123 优梦创客的官方博客:https://91make.top ...

  7. Unity进阶之ET网络游戏开发框架 07-修正游客登录的异步BUG

    版权申明: 本文原创首发于以下网站: 博客园『优梦创客』的空间:https://www.cnblogs.com/raymondking123 优梦创客的官方博客:https://91make.top ...

  8. Spring源码解析02:Spring IOC容器之XmlBeanFactory启动流程分析和源码解析

    一. 前言 Spring容器主要分为两类BeanFactory和ApplicationContext,后者是基于前者的功能扩展,也就是一个基础容器和一个高级容器的区别.本篇就以BeanFactory基 ...

  9. 【网络爬虫入门02】HTTP客户端库Requests的基本原理与基础应用

    [网络爬虫入门02]HTTP客户端库Requests的基本原理与基础应用 广东职业技术学院  欧浩源 1.引言 实现网络爬虫的第一步就是要建立网络连接并向服务器或网页等网络资源发起请求.urllib是 ...

随机推荐

  1. 20140117-配置文件为什么放在UI层

    配置文件为什么放在UI层 (刚才写着代码突然忘了配置文件为什么要放在UI层了,只记得晓虎老师强调过.找了半天视频……) 现总结一下: 晓虎老师给出的理由,大体如下:比如一个web项目,分成三层,DAL ...

  2. 20140115-SqlHelper为什么是静态的

    为什么SqlHelper(或工具类)是静态的? 静态构造函数仅调用一次(即只是在程序生命周期中实例一次),在程序驻留的应用程序域的生存期内,静态类一直保留在内存中 这样可以减少每次使用的实例过程,就是 ...

  3. 【超详细】vultr(CentOS7)+LNMP+WordPress 搭建个人博客

    心血来潮想搭建个人博客,我的vps只用来搭ss未免太过浪费 在这里记录一下搭建个人博客的历程 0x00 写在前面 why vultr: 优点:便宜. 性能优良.按时间计费(不用包年 学生党的福音).稳 ...

  4. backbone之路由锚点的替换

    1.需求 由于项目一开始做的时候不甚完善,所有的网页没有路由功能,导致一些搜索结果在页面跳转之后,没有被记录下来,在页面跳转之后回退,得到的是页面最原始的结果,没有指定的搜索条件.最近项目的在完善这些 ...

  5. MVC+EFCore 完整教程18 -- 升级分布视图至 View Component

    之前我们详细介绍过分布视图(partial view),在有一些更加复杂的场景下,.net core为我们提供了更加强大的组件 view  component, 可以认为view component是 ...

  6. 使用.net core基于Razor Pages开发网站一些工作笔记

    本文是在实践工作中遇到的一些问题记录,并给出是如何解决的,.net core已经升级到3.0版本了,其实在项目中很早就已经在使用.net core来开发后台接口了,正好有个网站项目,就使用了Razor ...

  7. python课堂整理10---局部变量与全局变量

    一.局部变量与全局变量 1. 没有缩进,顶头写的变量为全局变量 2. 在子程序里定义的变量为局部变量 3. 只有函数能把变量私有化 name = 'lhf' #全局变量 def change_name ...

  8. docker跨主机通信扁平化网络的设计与实现

    端口映射.ovs. fannel,weave 1.使用网桥实现跨主机容器连接 使用Open vSwitch实现跨主机容器连接

  9. 前端插件之Select2使用

    工欲善其事,必先利其器 本系列文章介绍我在运维系统开发过程中用到的那些顺手的前端插件,上一篇文章介绍了bootstrap-duallistbox这个插件的使用,这一篇开始Select2的征服之旅 Se ...

  10. 【iOS】"OS X"想要进行更改。键入管理员的名称和密码以允许执行此操作("OS X"想使用系统钥匙串)

    今天真机调试的时候遇到了这个问题,如下图: 每次调试都要输入两次用户名和密码,好麻烦的说…… 关键时刻找到了这篇文章:"Mac OS X"想要进行更改.键入管理员的名称和密码以允许 ...