理论参考:http://www.cnblogs.com/mezero/p/3955130.html
代码参考:http://www.omuying.com/article/78.aspx
背景:首先为什么会去了解对象池技术,原因是我的游戏在iOS上运行时出现了问题,有时会闪退,于是网上寻找解决方法,大神给出的答案是内存爆了。因为游戏中会频繁而且相对大量的创建( Instantiate )和( Destroy ),不仅导致游戏整个性能降低,而且如果创建的数量过多,内存很容易爆了。
目的:研究并实现对象池技术。
每次我都会先去看技术的理论知识,网上很多实现对象池技术的代码,但是理论就没有过多的介绍,因为对象池本身就很简洁形象。对象池技术就是:第一次创建的时候就将对象存储在一个池子中,当需要销毁的时候不直接销毁而是隐藏,当需要时在再次使用,就不需要再次实例化一个新的对象,直接把隐藏的对象显示并拿出来用。如果对象池中已经被隐藏的物体太久没有被重新使用,应该被真正的销毁。
池的最重要的特性,也就是对象池设计模式的本质是允许我们获取一个“新的”对象而不管它真的是一个新的对象还是循环使用的对象。
理论还是很容易理解的,估计有的人看了理论就可以自己去实现了~我没有那么大神,所以去网上找了很多代码资源自己看,去理解。接下来我把怎样实现一个对象池并应用于unity中的思路和代码写出来。
首先根据对象池概念,我们要怎样创建一个可以随时利用的对象池、怎样把实例化的物体放进对象池、怎样从对象池中取出对象、怎样实现超时的物体可以自动删除。
为了解决以上问题,首先我们要分好层级:对象池管理层(PoolManager),对象池组层(PoolItem),对象池组中成员层(PoolItemTime)。简单解释一下:
1:对象池管理层(PoolManager)这个很容易理解,即直接进行对象池操作的脚本。
2:对象池组层(PoolItem),为什么要分组呢?因为我们存进对象池同类型的东西可能不止一个,比如我生成一个Cube放进对象池中,如果在上一个Cube还没有被隐藏的时候我又需要一个,那么这个时候我必须从新生成一个Cube,因为对象池中并没有可以重复使用的对象(即被隐藏的Cube)。
3:对象池组中成员层(PoolItemTime),这个类的名字我是直接采用参考代码的名字,这个类就是用来管理单个物体的。
接下来:
1:我们要怎样创建一个可以随时利用的对象池,我们可以直接在对象池管理类(PoolManager)里面把要用的函数直接用Static修饰就可以了(类似于文档操作之类的),这样其他的脚本可以随时调用对象池操作。
2:怎样把实例化的物体放进对象池,我们需要Push()操作来完成,其中有一个关键性的问题就是,不同类的对象很容易管理,用一个Dictionary<T, T>就可以了,同类的怎么放在一起呢?仔细研究后发现自己思维受限,同类的物品直接用一个类似数组的东西存不就好了吗?下面的代码选择用Dictionary<int, GameObject>存起来,int是获取对象的HashCode存入,其实这个并没有什么用....跟数组的效果其实是一样的, 看3就知道为什么没用了。
3:怎样从对象池中取出对象,首先找到对象所在组,然后取组中第一个对象即可,知道为什么HashCode没用了吧?你也可以将2的int全部存成1。
解决上上面的问题其实整个对象池就没什么了,下面贴上代码,里面的注释和我的解释对应,应该很容易明白的。
PoolItemTime.cs
上面的代码可以直接用,用法是:
1:在每次需要Instantiate时先从对象池中获取出来,然后判断获取的是否为空,为空就Instantiate一个,然后放入对象池中管理。
2:每次需要destroy的时候不执行默认destroy而是执行Poolmanager中的没有真正删除的destroy(DestroyActiveObject)。
3:在GameManager的Update中执行超时检测(BeyondTimeObject)。
注意:
1:对象池中获取的对象需要必要的Init操作。
2:真正频繁删除复用的对象才加入对象池管理,否则并不能达到优化目的。
3:上代码没有对数量进行控制,可以自行修改添加。
4:对象池的应用Unity已经有插件可以使用(PoolManager插件),只需要使用功能的同学可以看:http://www.xuanyusong.com/archives/2974