每个需要进行资源管理的类都继承自IAssetManager,该类维护它所使用到的所有资源的一个资源列表。并且每个资源管理类可以重写其资源引用接口和解引用接口。

每个管理器有自己的管理策略,比如SceneManager对场景背景图可以保留最近使用的几张,使用LRU算法维护当前内存中的贴图张数等...

using UnityEngine;
using System.Collections;
using System.Collections.Generic; // 和资源有关的管理器都将继承自此类
public class IAssetManager
{
// 管理器所管理的资源列表,实际上是引用列表
protected List<string> lstRefAsset = new List<string>(); // 增加引用的资源
public virtual void RefAsset(string name)
{} // 以一定的策略卸载资源
public virtual bool UnloadAsset()
{ return true; } }

资源管理器类,UnloadUnusedAsset函数保证只有在真正有资源需要卸载的时候才调用Resources.UnloadUnusedAssets();

有一点需要注意的地方就是,在资源解压完成后,一定要记得要释放压缩包内存,即:

www.assetBundle.LoadAsync(GetAssetName(name), type);

www.assetBundle.Unload(false);

因为,www压缩包数据只能手动卸载,跳转场景都不会删除,所以加载完立即unload是一个好习惯。

还有一点需要注意的www.assetBundle.mainAsset是一个同步加载的操作,也就是说调用此函数时会卡。using UnityEngine;using System.Collections;

using UnityEngine;
using System.Collections;
using System.Collections.Generic; public class ResourceManager
{
// 已解压的Asset列表 [prefabPath, asset]
private Dictionary<string, Object> dicAsset = new Dictionary<string, Object>();
// "正在"加载的资源列表 [prefabPath, www]
private Dictionary<string, WWW> dicLoadingReq = new Dictionary<string, WWW>(); public Object GetResource(string name)
{
Object obj = null;
if (dicAsset.TryGetValue(name, out obj) == false)
{
Debug.LogWarning("<GetResource Failed> Res not exist, res.Name = " + name);
if (dicLoadingReq.ContainsKey(name))
{
Debug.LogWarning("<GetResource Failed> The res is still loading");
}
}
return obj;
} // name表示prefabPath,eg:Prefab/Pet/ABC
public void LoadAsync(string name)
{
LoadAsync(name, typeof(Object));
} // name表示prefabPath,eg:Prefab/Pet/ABC
public void LoadAsync(string name, System.Type type)
{
// 如果已经下载,则返回
if (dicAsset.ContainsKey(name))
return; // 如果正在下载,则返回
if (dicLoadingReq.ContainsKey(name))
return; // 添加引用
RefAsset(name);
// 如果没下载,则开始下载
CoroutineProvider.Instance().StartCoroutine(AsyncLoadCoroutine(name, type));
} private IEnumerator AsyncLoadCoroutine(string name, System.Type type)
{
string assetBundleName = GlobalSetting.ConvertToAssetBundleName(name);
string url = GlobalSetting.ConverToFtpPath(assetBundleName);
int verNum = GameApp.GetVersionManager().GetVersionNum(assetBundleName); Debug.Log("WWW AsyncLoad name =" + assetBundleName + " versionNum = " + verNum);
if (Caching.IsVersionCached(url, verNum) == false)
Debug.Log("Version Is not Cached, which will download from net!"); WWW www = WWW.LoadFromCacheOrDownload(url,verNum);
dicLoadingReq.Add(name, www);
while (www.isDone == false)
yield return null; AssetBundleRequest req = www.assetBundle.LoadAsync(GetAssetName(name), type);
while (req.isDone == false)
yield return null; dicAsset.Add(name, req.asset);
dicLoadingReq.Remove(name);
www.assetBundle.Unload(false);
www = null;
// Debug.Log("WWW AsyncLoad Finished " + assetBundleName + " versionNum = " + verNum);
} public bool IsResLoading(string name)
{
return dicLoadingReq.ContainsKey(name);
} public bool IsResLoaded(string name)
{
return dicAsset.ContainsKey(name);
} public WWW GetLoadingWWW(string name)
{
WWW www = null;
dicLoadingReq.TryGetValue(name, out www);
return www;
} // 移除Asset资源的引用,name表示prefabPath
public void UnrefAsset(string name)
{
dicAsset.Remove(name);
} private string GetAssetName(string ResName)
{
int index = ResName.LastIndexOf('/');
return ResName.Substring(index + , ResName.Length - index - );
} public void UnloadUnusedAsset()
{
bool effectNeedUnload = GameApp.GetEffectManager().UnloadAsset();
bool worldNeedUnload = GameApp.GetWorldManager().UnloadAsset();
bool sceneNeedUnload = GameApp.GetSceneManager().UnloadAsset();
if (effectNeedUnload || worldNeedUnload || sceneNeedUnload)
{
Resources.UnloadUnusedAssets();
}
} // 根据资源路径添加资源引用,每个管理器管理自己的引用
private void RefAsset(string name)
{
// 模型之类的
if (name.Contains(GlobalSetting.CharacterPath))
GameApp.GetWorldManager().RefAsset(name);
// 图片之类的
else if (name.Contains(GlobalSetting.TexturePath))
GameApp.GetUIManager().RefPTexture(name);// 特效之类的
else if (name.Contains(GlobalSetting.EffectPath))
GameApp.GetEffectManager().RefAsset(name);
     ......
     else
Debug.LogWarning("<Res not ref> name = " + name);
} }

资源管理的关键在于以下几点:

(1)资源所对应的几块内存的管理,Unity的内存一直是一个相对比较棘手的方面,所以一定要多做尝试找到规律和方法;

(2)资源加载、卸载策略,什么时候加载什么时候卸载需要根据游戏类型来进行定制;

(3)资源打包策略,也就是以什么单位进行什么类型的资源打包,原则上是让同一资源尽量只需要打包一次,比如多个场景都用到了同一棵树,那么最好是对这棵树单独打包等等。

    比如Prefab1和Prefab2同时引用了Fbx1,将2个prefab单独打包时都会分别包含Fbx1,并且解压到内存时,也会有2份独立的Fbx1,这样会造成内存变大,这点一定要注意。

......

Unity内存和资源这一块虽然显得比较拖泥带水,不过只要使用的够规范,一般还是能够保证内存的干净的。

......

还有一个尚未解决的问题,Unity使用www方式加载资源的时候,不能进行同步加载操作,只能异步,我见到过其他人也遇到过这个问题,很是蛋疼菊紧。

AssetBundle系列——资源的加载、简易的资源管理器的更多相关文章

  1. 在Unity3D的网络游戏中实现资源动态加载

    用Unity3D制作基于web的网络游戏,不可避免的会用到一个技术-资源动态加载.比如想加载一个大场景的资源,不应该在游戏的开始让用户长时间等待全部资源的加载完毕.应该优先加载用户附近的场景资源,在游 ...

  2. unity3d进行脚本资源打包加载

    原地址:http://www.cnblogs.com/hisiqi/p/3204752.html 本文记录如何通过unity3d进行脚本资源打包加载 1.创建TestDll.cs文件 public c ...

  3. (转)在Unity3D的网络游戏中实现资源动态加载

    原文:http://zijan.iteye.com/blog/911102 用Unity3D制作基于web的网络游戏,不可避免的会用到一个技术-资源动态加载.比如想加载一个大场景的资源,不应该在游戏的 ...

  4. Unity开发实战探讨-资源的加载释放最佳策略简要心得

    Unity开发实战探讨-资源的加载释放最佳策略简要心得 看过我另外一篇关于Unity资源释放随笔<Unity开发实战探讨-资源的加载释放最佳策略>如果觉得略微复杂,那么下面是一些比较简要的 ...

  5. Unity开发实战探讨-资源的加载释放最佳策略

    注:本文中用到的大部分术语和函数都是Unity中比较基本的概念,所以本文只是直接引用,不再详细解释各种概念的具体内容,若要深入了解,请查阅相关资料. Unity的资源陷阱 游戏资源的加载和释放导致的内 ...

  6. 老调重弹:JDBC系列之<驱动加载原理全面解析) ----转

      最近在研究Mybatis框架,由于该框架基于JDBC,想要很好地理解和学习Mybatis,必须要对JDBC有较深入的了解.所以便把JDBC 这个东东翻出来,好好总结一番,作为自己的笔记,也是给读者 ...

  7. VC++ 使用WebBrowser控件中html文件以资源形式加载

    . . . . //加载资源文件中的HTML,IDR_HTML1就是HTML文件在资源文件中的ID wchar_t self_path[MAX_PATH] = { }; GetModuleFileNa ...

  8. 关于cocos2d-x 与 cocos2d-html5 资源预加载的思考

    移动端资源预加载,可以做到需要加载的时候,从本地磁盘加载到内存,当纹理不需要的时候,都是强制清理内存里的纹理占用: cc.TextureCache.getInstance().removeAllTex ...

  9. 资源预加载 Preload

    当提到前端性能优化时,我们首先会联想到文件的合并.压缩,文件缓存和开启服务器端的 gzip 压缩等,这使得页面加载更快,用户可以尽快使用我们的 Web 应用来达到他们的目标. 资源预加载 是另一个性能 ...

随机推荐

  1. 无线局域网络 WIFI/WAPI/WLAN区别浅析

    WIFI和WAPI的区别 既然WIFI和WAPI都是WLAN的传输协议,那么两者究竟都有怎样的区别? 首先第一点区别在于,两者的缔造者不一样.WIFI是又国外制定的一个协议,而WAPI是由中国制定的, ...

  2. django文件上传下载

    views: def mgmt_files(request): #列出树形目录,上传文件页面 if request.method == 'POST': path_root = "D:\\py ...

  3. Swift入门篇-结构体

    前面主要是介绍swift语言中基本类型的用法,今天给大家介绍的是swift的结构体的用法,swift中结构体的用法和其他语言的用法,还有不太一样,不过您多敲几遍,就可以理解结构体,结构体在ios开发中 ...

  4. 袭击Mercurial SCM(HG)

    这个叫水银的源代码管理工具尽管默默无闻,但还是得到了非常多团队的使用. 为了迎合某些团队的须要,我们也要用它来管理我们的代码. 今天的任务是先袭击学习.磨刀不误砍柴工. 对工具的掌握越快.工作的效率就 ...

  5. C#新开一个线程取到数据,如何更新到主线程UI上面

       一:问题 之前有被面试官问过,在WinForm中,要去网络上获取数据,由于网络环境等原因,不能很快的完成,因此会发生进程阻塞,造成主进程假死的现象,需要怎么解决?    二:思路 因此,往往是新 ...

  6. Swift:如何判断一个对象是否是某个类(或其子类)的实例

    在OC中我们直接可以用如下方法即可 [obj  isKindOfClass:[obj class]]; 在Swift中,并没有 .class 属性或者方法, 便可以用如下方法 class Person ...

  7. 在GridView中使用radioButoon

    在GridView中使用radioButoon 方法一: <input type="radio" id='radioSelectFeed' name="radioD ...

  8. linux系统和依赖包常用下载地址

    http://ftp.gnome.org/pub/gnome/sources/gstreamer/0.10/ http://www.linuxfromscratch.org/blfs/view/svn ...

  9. [转]HTTP请求模型和头信息

    原文链接:http://www.java3z.com/cwbwebhome/article/article2/2406.html 目录 一.连接至Web服务器 二.发送HTTP请求 三.服务端接受请求 ...

  10. SQLSERVER吞噬内存解决记录

    现在手上有一个不大不小的系统,运行了一段时间,因为是24*7不断运行,所以内存逐渐增高,慢慢的会飙到95%以上,然后不得不重启电脑,因为用的是云,怕虚拟机重启down掉起不来,重启操作还只能在凌晨4. ...