Unity的资源管理是怎样的


这篇文章将为大家详细讲解有关Unity的资源管理是怎样的,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文章后对相关知识有一定的了解。在Unity最佳实践明确指出, 要使用AssetBundle而不是Resources目录来管理资源。然而,事情并不像Unity官方描述的那么美好。因为使用AssetBundle我们甚至无法实现一个易用的,完备的资源管理方案。据Unity官方说,一般有两种方案。方案一,如果你的游戏是关卡性质的,可以在一个关卡里加载所有AssetBundle,然后在进入下一关卡时,卸载本关卡中加载的所有AssetBundle. 但这种机制似乎只对愤怒的小鸟这种小游戏才适用吧:D。方案二,如果你的游戏不是关卡类的,那么Unity推荐做一个资源对AssetBundle引用计数。如果一个对象(Asset或其他AssetBundle)引用此AssetBundle则其引用计数加1. 如果此AssetBundle首次加载(即加载前引用计数为0), 还需要递归对其依赖引用计数加1。如果一个AssetBundle的引用计数为0则释放这个AssetBundle,同时还需要递归对其依赖引用计数减1.除非,我们做像愤怒小鸟一样的通关游戏,不然似乎只有方案二给我们用。而且方案二乍一看是完备的,因为这正是GC算法的一种实现。但是如果稍微仔细思考一下就会发现,这个方案只是AssetBundle的管理方案,是个半成品,要如何管理管理资源之间的依赖,Unity却只字未掉,看起来是让用户自己想办法,这似乎与其易学易用的宗旨不太相符。下面来分析一下Unity中资源之间的关系。
在Unity中资源大约分为以下几种:纹理(Texture)、网格(Mesh)、动画片段(AnimationClip)、音频片段(AudioClip)、材质(Material)、着色器(Shader)、字体资源(Font)以及文本资源(TextAsset)。AssetBundle中还有一个极其特殊的存在,那就是Prefab, AssetBundle.LoadAsset时返回的是GameObject, 但是又必须经过Instantitate之后变成另外一个GameObject才能使用。此后所说的GameObject均是Instantitate之后的GameObject。GameObject可以添加各种Component来引用上述除资源,还可以通过代码动态增减某个GameObject上的Component或者修改Component对资源的引用。这种灵活性给资源管理带来了巨大麻烦,而没有这种灵活性,逻辑的实现就会更麻烦。下面,举例来说明一下,要正确管理GameObject和资源之间的引用关系有多么艰难。Prefab P能过Instantitate生成A,B,C,D四个GameObject.执行如下代码之后,A引用{P,T1}, B引用{P,T1}, C引用{P,T3}。并且T2应该被Unload。1: A.GetComponent().sprite = (Sprite)T1;
2: B.GetComponent().sprite = (Sprite)T1;
3: C.GetComponent().sprite = (S 香港云主机prite)T2;
4: C.GetComponent().sprite = (Sprite)T3;
要想自动正确的管理GameObject和资源的引用关系,就必须要感知到对GameObject的赋值操作。例如:所有的sprite赋值都必须使用类似SpriteAssign(SpriteRender sr, Sprite s)的接口。SpriteAssign的执行流程通常是这样的。检查sprite的值是不是T1相同,如果是相同则不做处理检查sprite的值是不是从P中clone过来的,如果不是,将此sprite的引用计数减1将T1的引用计数加1
如果P是一个树状态结构,即有P–(child)–>p1–(child)–>p2。1: A.p1.p2.GetComponent().sprite = (Sprite)T1;
2: B.p1.p2.GetComponent().sprite = (Sprite)T1;
3: C.p1.p2.GetComponent().sprite = (Sprite)T2;
4: C.p1.p2.GetComponent().sprite = (Sprite)T3;
SpriteAssign接口中的步骤2就显得格外复杂,它必须修正引用关系如下:A引用{P,T1}, B引用{P,T1}, C引用{P,T3}。同时Destory操作也要被感知,如果Destory(A)则需要释放A引用的资源,而如果Destory(A.p1.p2)则需要修正A对资源的引用情况。因为此时的引用关系是,A引用{P}。换句话说Destroy的开销也会变大。而赋值和Destory都算不上低频操作,尤其是赋值操作。这样的开销已经足够让程序慢上好几倍了。如果不能承受这些开销,全自动化资源管理是不可能实现的。我想这也是Unity不默认提供一套标准的全自动化资源管理方案的根本原因吧。受方案一的启发,我觉得可以通过如下接口做一个半自动化的资源管理器。每一个level对象都会记录在level.open()和level.close()之间所有加载过的资源,加过载多少次就记录多少次,这些资源会在执行level.dispose()时如实的进行释放。其中stack在管理UI资源方面几乎已经达到了全自动化,当你打开一个UI时调用stack.push,在退出此UI时调用stack.pop会自动释放在此UI期间你所加载的全部资源。而在其他不具有栈式加载资源特征的地方,level类也提供了一种方便的半自动化管理方案。最重要的是,此种方案的开销和复杂度,都要远低于全自动化管理方案。关于Unity的资源管理是怎样的就分享到这里了,希望以上内容可以对大家有一定的帮助,可以学到更多知识。如果觉得文章不错,可以把它分享出去让更多的人看到。

相关推荐: 笔记本电脑如何设置合上盖子不进入睡眠模式

这篇文章将为大家详细讲解有关笔记本电脑如何设置合上盖子不进入睡眠模式,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。1.在键盘上按下【win+i】组合键,调出设置选项界面,选择【系统】进入。2.进入后在左侧选择【电源和睡眠】,…

免责声明:本站发布的图片视频文字,以转载和分享为主,文章观点不代表本站立场,本站不承担相关法律责任;如果涉及侵权请联系邮箱:360163164@qq.com举报,并提供相关证据,经查实将立刻删除涉嫌侵权内容。

(0)
打赏 微信扫一扫 微信扫一扫
上一篇 08/13 16:35
下一篇 08/13 16:36

相关推荐