Unity3d:特效对象池,超时删除池内GameObject,GC权值

技术要点:

  1. 不用的粒子返回缓冲池
  2. 按照单个粒子名字创建一个池,当a池上一次操作时间(回收对象)超过阈值,每隔一段时间删除池里一个GameObject,可针对单个粒子配表,频繁使用的,驻留时间更长
  3. 一个完整池被删除,增加GC权值,GC权值达到最大值调用System.GC.Collect()

遇到问题:

  1. 父节点OnDisable,不可设置子节点的父物体
    Cannot set the parent of the GameObject ''XXX“ while activating or deactivating the parent GameObject “XXX” ,由于父对象进入回收池时,代码正在对该对象子节点的父节点进行修改。修改方式为:给子节点套一层空对象
  2. 回收后再从池里调出,粒子只显示一半,或者不显示
    每次粒子取出后,如果上一个粒子是在scroll中使用,在裁剪shader影响下,会出现裁剪一半情况,要重新设置粒子裁剪区
        public void ResetMaskParams(Material mat)
        {
            mat.SetFloat("_MinX", -1);
            mat.SetFloat("_MinY", -1);
            mat.SetFloat("_MaxX", 1);
            mat.SetFloat("_MaxY", 1);
        }
  1. 有些粒子是带有特殊处理,例如用完即销毁子物体;或者挂载脚本会丢失引用,不放入缓冲池

代码

得到粒子

简单测试,真正使用用resid代替objPrefab,并用Assetbundle同步加载

public GameObject GetEffect(GameObject objPrefab)
        {
            string name = GetNoCloneName(objPrefab.name);
            GameObject obj = null;
            if (m_dicPool.ContainsKey(name))
            {
                if (m_dicPool[name] != null && m_dicPool[name].Count > 0)
                {
                    obj = m_dicPool[name].Dequeue();
                    while (obj == null && m_dicPool[name].Count > 0)
                    {
                        obj = m_dicPool[name].Dequeue();
                        if (obj != null)
                        {
                            //return obj;
                            break;
                        }
                    }
                    if (obj == null)
                    {
                        obj = GameObject.Instantiate(objPrefab) as GameObject;
                    }

                }
                else
                {
                    obj = GameObject.Instantiate(objPrefab) as GameObject;
                }
            }
            else
            {
                m_dicPool[name] = new Queue<GameObject>(m_count);
                obj = GameObject.Instantiate(objPrefab) as GameObject;
            }
            obj.SetActive(true);
            SetInDicUse(obj);
            return obj;
        }

回收粒子

public void RecycleEffect(GameObject obj)
        {

            if (obj == null)
            {
                return;
            }
            
            //这里加要过滤不回收的粒子名字,或条件
            string name = GetNoCloneName(obj.name);


            m_lastUsedTime[name] = Time.time;
            SetOutDicUse(obj);
            obj.transform.SetParent(GetEffectPoolObj());
            obj.SetActive(false);

            if (m_dicPool.ContainsKey(name))
            {
                if (m_dicPool[name] == null)
                {
                    m_dicPool[name] = new Queue<GameObject>(m_count);
                }
                m_dicPool[name].Enqueue(obj);
            }
            else
            {
                m_dicPool[name] = new Queue<GameObject>(m_count);
                m_dicPool[name].Enqueue(obj);
            }
        }

粒子池超时未再使用,Destroy一个GameObject

private void FixedUpdate()
{
foreach (var item in m_lastUsedTime)
                {
                    string keyName = item.Key;
                    float lastTime = item.Value;
                    if (m_dicUse.ContainsKey(keyName) && m_dicUse[keyName].Count == 0)
                    {
                        if (m_dicPool.ContainsKey(keyName))
                        {
                            if (m_dicPool[keyName].Count > 0)//某个池里还有空闲对象
                            {
                                if (Time.time - lastTime > GetDeleOneObjTimeClip(m_dicPool[keyName].Count)) // 超时移除一个空闲对象
                                {
                                    GameObject idleObj = m_dicPool[keyName].Dequeue();
                                    GameObject.Destroy(idleObj);
                                    timeUpdateList.Add(keyName);
                                }
                            }
                            else//某个池里无空闲对象
                            {
                                if (Time.time - lastTime > mResReleaseTime)
                                {
                                    releaseList.Add(keyName);
                                }

                            }
                        }
                    }
                }


                for (int i = 0; i < timeUpdateList.Count; i++)
                {
                    m_lastUsedTime[timeUpdateList[i]] = Time.time; // 上次删除缓冲池里某个obj的时间
                }

GC权值

public void AddUnloadWeights(int nWeights = 1)
        {
            m_nSumWeights += nWeights;
        }

        void Update()
        {
            //每60帧执行一次检测
            if (Time.frameCount % CHECK_INTERVAL_FRAME == 0)
            {
                TryUnloadUnusedAssets();
            }
        }

        void TryUnloadUnusedAssets()
        {
            if ((Time.realtimeSinceStartup - m_fLastUnloadTime >= m_nMaxInterval)
                || (m_nSumWeights >= m_nWeightsThreshold))
            {
                DoUnloadUnusedAssets();
            }
        }

        void DoUnloadUnusedAssets()
        {
            Resources.UnloadUnusedAssets();
            System.GC.Collect();
            m_nSumWeights = 0;
            m_fLastUnloadTime = Time.realtimeSinceStartup;

        }

源码

https://github.com/luoyikun/UnityForTest
EffectPoolMgr.cs

  • 2
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
Unity中,虽然没有内置的对象类,但是可以通过自定义脚本来实现对象的功能。下面是一个简单的示例代码,展示了如何创建一个基本的对象: ```csharp using System.Collections.Generic; using UnityEngine; public class ObjectPool : MonoBehaviour { public GameObject prefab; // 预制体 public int poolSize; // 对象大小 private List<GameObject> objectPool; void Start() { objectPool = new List<GameObject>(); for (int i = 0; i < poolSize; i++) { GameObject obj = Instantiate(prefab); obj.SetActive(false); objectPool.Add(obj); } } public GameObject GetObjectFromPool() { for (int i = 0; i < objectPool.Count; i++) { if (!objectPool[i].activeInHierarchy) { objectPool[i].SetActive(true); return objectPool[i]; } } // 如果对象中没有可用对象,则动态创建一个新对象 GameObject newObj = Instantiate(prefab); objectPool.Add(newObj); return newObj; } public void ReturnObjectToPool(GameObject obj) { obj.SetActive(false); } } ``` 在上述代码中,我们创建了一个名为ObjectPool的脚本。在Start方法中,我们初始化了对象,根据poolSize的来创建一定数量的对象,并将它们存储在objectPool列表中。 GetObjectFromPool方法用于从对象中获取一个可用的对象。我们遍历objectPool列表,寻找第一个处于非激活状态的对象,并将其设置为激活状态,然后返回该对象。如果对象中没有可用对象,则动态创建一个新对象,并将其添加到objectPool列表中。 ReturnObjectToPool方法用于将使用完的对象返回到对象中。我们将传入的对象设置为非激活状态,以便下次复用。 通过这样的自定义脚本,你可以在Unity中实现对象的功能,以提高游戏的性能和效率。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

四夕立羽

你的鼓励将是我创作的最大动力。

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值