对象池的出现,主要是因为Instantiate和Destroy方法开销比较大,但一些游戏物体,比如子弹等,会频繁地出现,并很快销毁,如果每次发射一枚子弹都进行一次实例化,每次击中后又都进行一次销毁,那对游戏的性能开销就不能忽视,对象池就因此而生。

我的对象池的思路是:在需要对象时,先看对象池里有没有,如果有的话就直接取出来并显示,没有的话就实例化一个;在原本需要摧毁对象时,改为开销较小的将对象取消显示的方式,放到对象池里等待下一次使用。

对象池应该能在需要使用的地方直接调用,并且本身只需要存在一个,因此显然应该使用单例模式。主要有两个方法:GetObject 和 PushObject,分别将对象从池子里取出和放回。对象池中对象的存储应该使用字典的方式,键为string类型,表示对象的名字,值为Queue<GameObject>类型,表示一组对象组成的队列。

GetObject方法思路:首先判断对象字典中有没有以所需对象为名字的键,以及这个键所对应的队列中是否有可用的对象(Count不为0),如果满足,那直接取出队首并让其显示;如果不满足,就实例化一个并放进池中。当不满足条件时,有可能是这个所需物体从来没有被实例化过,因此这时候需要建立一个以该对象为名的空物体,放到Pool物体下,再将实例化的对象放到这个空物体下,这样便于在场景中查看和管理对象池中的情况。

PushObject方法思路:首先将要回收的对象名字中的(Clone)替换为空串,因为如果是实例化出的对象,后面会自动加一个(Clone)后缀,而字典中的键是不带(Clone)的,所以先替换为空串。之后检测字典中是否有以该对象为名的键,如果没有,就新建立一个键值对,之后将所回收对象放进队列并取消显示即可。

代码如下:

using System.Collections;
using System.Collections.Generic;
using UnityEngine;

public class ObjectPool
{
    private static ObjectPool instance;

    private Dictionary<string, Queue<GameObject>> objectPool = new Dictionary<string, Queue<GameObject>>();

    private GameObject pool;
    public static ObjectPool Instance
    {
        get
        {
            if (instance == null)
            {
                instance = new ObjectPool();
            }
            return instance;
        }
    }

    public GameObject GetObject(GameObject prefab)
    {
        GameObject gameObject;

        if (!objectPool.ContainsKey(prefab.name) || objectPool[prefab.name].Count == 0)
        {
            gameObject = GameObject.Instantiate(prefab);
            PushObject(gameObject);
            if (pool == null)
            {
                pool = new GameObject("ObjectPool");
            }
            GameObject childPool = GameObject.Find(prefab.name + "Pool");
            if (!childPool)
            {
                childPool = new GameObject(prefab.name + "Pool");
                childPool.transform.SetParent(pool.transform);
            }
            gameObject.transform.SetParent(childPool.transform);
        }
        gameObject = objectPool[prefab.name].Dequeue();
        gameObject.SetActive(true);
        return gameObject;
    }

    public void PushObject(GameObject prefab)
    {
        string objectName = prefab.name.Replace("(Clone)", string.Empty);
        if (!objectPool.ContainsKey(objectName))
            objectPool.Add(objectName, new Queue<GameObject>());
        objectPool[objectName].Enqueue(prefab);
        prefab.SetActive(false);
    }

}

之后在原本需要实例化某个prefab的地方,直接ObjectPool.Instance.GetObject(prefab),这个方法有返回值,可以定义一个游戏物体并给它设置位置旋转等信息。在原本需要摧毁某个prefab的地方,直接ObjectPool.Instance.PushObject(prefab) 即可。