欢迎转载,请注明作者【燕良@游戏开发】及原文地址:http://blog.csdn.net/neil3d/article/details/39580197

另外。欢迎大家来我的QQ群交流各种游戏引擎相关的技术:游戏引擎能吃吗(264656505)

话说,经过这段时间的学习和摸索。对于Unity3D的开发思路已经基本清晰了。

只有还剩下一个AssetBundle机制还没有搞透,这个涉及到前期项目的资源规划、资源管理代码的写法。以及自己主动更新机制的实现。

所以,还是想先把游戏逻辑的进一步分析押后,先来看一下《炉石传说》Asset管理。必须得说一下的是,眼下分析都是PC版的程序集。对于移动端不一定全然合适,且当做一个案例分析吧。

本文主要讲述《炉石传说》的AssetBundle的管理机制。它的机制比較简单清晰,中规中矩,中间的分析过程就不讲了,直接展现其架构设计和代码逻辑组织。

先从Asset管理相关的类讲起。

class Asset :资源信息描写叙述


Asset类,并无论理直接的资源对象,而是保存的一个Asset相关的信息。详细请看上图。
另外,它另一个“paths”变量,这是一个Dictionary。key是AssetFamily枚举。value是Assetbundle的路径和资源路径。

以下的AssetFamily一节详解。

enum AssetFamily - 资源分类


如上图所看到的:
  • 炉石依据资源的不同类型进行分别的AssetBundle打包,一类资源相应一个或者多个资源包。(一类资源分多个包的规则不得而知);
  • 有的资源包真的本地化单独打包,比如“fonts0.unity3d”==》“fontszhCN0.unity3d”;
  • 在程序中。资源包的分类相应枚举类型“AssetFamily”。
  • 资源包的详细路径信息。存储在Asset.paths,这是一个静态变量;在初始化时,手动填写必要的信息,类似这样:
        Dictionary<AssetFamily, AssetFamilyPathInfo> dictionary = new Dictionary<AssetFamily, AssetFamilyPathInfo>();
    AssetFamilyPathInfo info = new AssetFamilyPathInfo {
    format = "Data/Actors/{0}.unity3d",
    sourceDir = "Assets/Game/Actors"
    };
    info.exts = new string[] { "prefab" };
    dictionary.Add(AssetFamily.Actor, info);
  • 另外。另一个class AssetBundleInfo是记录了每种AssetBundle相应的主文件名称。以及包文件的个数、相应的对象类型等信息;详见下图:

class AssetLoader :资源载入


游戏执行时须要载入各种资源。基本上都是通过AssetLoader(也有个别情况适用了Resources.Load())。接下来我们就重点看一下AssetLoader的实现思路。


AssetLoader对上层提供资源对象载入接口。对于每种类型的资源都提供一组函数,比如LoadCardPrefab,LoadActor等等。对于对象载入完毕、载入进度等提供回调函数。这些函数仅仅是一些简单的包装。其内部都调用到LoadCachedGameObject()或LoadCachedObject()这两个核心函数。
从这两个函数的流程能够看到,资源载入使用到了Cache机制:
  • 首先从AssetCache中查找,假设找到了。则更新Cache项的时间戳。并调用回调;
  • 假设没有找到,则向AssetCache加入一个Request,然后启动Coroutine:CreateCachedAsset(),它的调用步骤是:
    • 调用AssetCache.StartLoading()。
    • 启动Coroutine:CreateCachedAsset_FromBundle<RequestType>():
      • 使用AssetLoader.GetBundleForAsset()找到资源所属的AssetBundle;
      • 调用AssetBundle.LoadAsync()来真正载入资源;
      • 在载入的过程中,依据处理的结果调用:AssetCache.CacheRequest的OnLoadFailed()、OnLoadSucceeded()、OnProgressUpdate()等函数;
    • 在AssetCache查找此资源。假设找到了,则载入成功,调用回调函数;

      调用AssetCache.StopLoading();

我们都知道在开发过程中,不能使用AssetBundle(每次启动都要打包。肯定收不了)。

怀疑它的Editor模式相关的代码是用预编译宏处理来实现的,所以未出如今公布出来的程序集其中。类似这样:

#if UNITY_EDITOR
Obj = Resources.LoadAssetAtPath(assetPath, typeof(T));
if (Obj == null)
Debug.LogError ("Asset not found at path: " + assetPath);
yield break;
#else

class AssetCache :资源的Cache机制

前面在AssetLoader一节我们已经讲到了AssetCache机制,这里再做一个具体的阐述。


前面我们已经讲到:
  • AssetCache中的资源项的时间戳,由AssetLoader在资源载入请求时维护;
  • AssetCache主要负责管理Cache数据,而真正的资源载入动作还是在AssetLoader中运行。
AssetCache的资源淘汰主要由外部的各个模块依据自己觉得须要的时机去调用,比如:
  • SceneMgr.ClearCachesAndFreeMemory()
  • LoadingScreen.ClearAssets()
  • SoundMgr.UnloadSoundBundle()
  • 等等
另外。程序启动时会自己主动更新资源包(在Login.OnAssetsVersion()中启动),主要是通过UpdateManager和Downloader两个类来处理。
OK,总结一下炉石的资源管理机制:
  • 对游戏资源依照类型分包。每一类资源包能够有多个。
  • 在游戏执行时使用Cache机制;
最后,还是顺便炫一下战绩:

《炉石传说》架构设计赏析(4):Asset管理的更多相关文章

  1. DotNET应用架构设计指南 安全 运行管理和通讯策略

    DotNET应用架构设计指南(第三章:安全 运行管理和通讯策略(13-16)) 安全 运行管理和通讯策略 组织策略定义的规则是支配应用程序如何安全,如何管理,不同的应用程序组件是如何和另一组件及外部服 ...

  2. 【tornado】系列项目(一)之基于领域驱动模型架构设计的京东用户管理后台

    本博文将一步步揭秘京东等大型网站的领域驱动模型,致力于让读者完全掌握这种网络架构中的“高富帅”. 一.预备知识: 1.接口: python中并没有类似java等其它语言中的接口类型,但是python中 ...

  3. 炉石传说 C# 设计文档(序)

    经过3个月的开发,有很多感触. 以前一直以为技术是开发成败的第一因素,现在发现,等到你代码写的时间够长,经验够丰富,什么功能都能随手完成,对于业务的分析能力变成了第一位. 炉石山寨版的BS版本用到的H ...

  4. 【Python之路】特别篇--基于领域驱动模型架构设计的京东用户管理后台

    一.预备知识: 1.接口: - URL形式 - 数据类型 (Python中不存在) a.类中的方法可以写任意个,想要对类中的方法进行约束就可以使用接口: b.定义一个接口,接口中定义一个方法f1: c ...

  5. 搭建rtmp直播流服务之3:java开发ffmpeg实现rtsp转rtmp并实现ffmpeg命令的接口化管理架构设计及代码实现

    上一篇文章简单介绍了java如何调用ffmpeg的命令:http://blog.csdn.net/eguid_1/article/details/51777716 上上一篇介绍了nginx-rtmp服 ...

  6. 架构设计之Spring-Session的分布式集群会话管理

    发表于 2017-04-24  |  160次围观   |   分类于 架构设计   |   暂无评论 前言 通常在web开发中,回话管理是很重要的一部分,用于存储与用户相关的一些数据.对于JAVA开 ...

  7. 架构设计 | 分布式系统调度,Zookeeper集群化管理

    本文源码:GitHub·点这里 || GitEE·点这里 一.框架简介 1.基础简介 Zookeeper基于观察者模式设计的组件,主要应用于分布式系统架构中的,统一命名服务.统一配置管理.统一集群管理 ...

  8. 架构设计 | 接口幂等性原则,防重复提交Token管理

    本文源码:GitHub·点这里 || GitEE·点这里 一.幂等性概念 1.幂等简介 编程中一个幂等操作的特点是其任意多次执行所产生的影响均与一次执行的影响相同.就是说,一次和多次请求某一个资源会产 ...

  9. 架构设计 | 基于Seata中间件,微服务模式下事务管理

    源码地址:GitHub·点这里 || GitEE·点这里 一.Seata简介 1.Seata组件 Seata是一款开源的分布式事务解决方案,致力于提供高性能和简单易用的分布式事务服务.Seata将为用 ...

随机推荐

  1. 如何修改MFC的图标

    原文:如何修改MFC的图标 修改左上角的图标和任务栏里图标 在对话框构造函数中 CTestDlg::CTestDlg(CWnd* pParent /*=NULL*/) : CDialog(CTestD ...

  2. 07CSS表格

    CSS表格 表格边框 如需在 CSS 中设置表格边框,请使用 border 属性. table, th, td { border: 1px solid blue; } 折叠边框 border-coll ...

  3. No-7.运算符

    数学符号表链接:https://zh.wikipedia.org/wiki/数学符号表 01. 算数运算符 是完成基本的算术运算使用的符号,用来处理四则运算 运算符 描述 实例 + 加 10 + 20 ...

  4. No-3.算数运算符

    01. 算数运算符 算数运算符是 运算符的一种 是完成基本的算术运算使用的符号,用来处理四则运算 运算符 描述 实例 + 加 10 + 20 = 30 - 减 10 - 20 = -10 *  乘 1 ...

  5. getQueryString(option)的用法

    //页面参数接收1.function getQueryString(name) { var reg = new RegExp("(^|&)" + name + " ...

  6. BZOJ4873 LuoguP3749 寿司餐厅

    题面太长,请诸位自行品尝—>寿司餐厅 分析: 首先题目中给了限制条件,假如选了D(i,j)(i<j),那么也就选了D(i+1,j)和D(i,j-1)两个点. 于是我们一下就明白了,哦,最大 ...

  7. 秋招复习-C++( 一)

    Linux/Unix编程部分 1.进程间通信方式:信号,信号量,消息队列,共享内存,套接字Socket 2.ipcs: Linux/Unix下的命令,可以用来查看当前系统中所使用的进程间通信方式的各种 ...

  8. UART整理

    通用异步收发器简称UART,英文全称"Universal Asynchronous Receiver Transmitter".UART使用标准的TTL/CMOS逻辑电平(0~5V ...

  9. 【Codeforces 1117C】Magic Ship

    [链接] 我是链接,点我呀:) [题意] 题意 [题解] 我们可以把这个行船的过程分解成两个过程 1.船经过时间t被风吹到了某个地方 2.船用这t时间尝试到达终点(x2,y2) 会发现如果时间t能最终 ...

  10. Android渲染器Shader:LinearGradient(一)

     Android渲染器Shader:LinearGradient(一) LinearGradient是Android的线性渲染器.我写5个LinearGradient渲染器渲染后的View表现结果 ...