【Unity程序技巧】加入缓存池存储地图资源,节省资源,避免多次CG


👨‍💻个人主页@元宇宙-秩沅

👨‍💻 hallo 欢迎 点赞👍 收藏⭐ 留言📝 加关注✅!

👨‍💻 本文由 秩沅 原创

👨‍💻 收录于专栏Unity基础实战

⭐🅰️⭐



文章目录

    • ⭐🅰️⭐
    • ⭐前言⭐
    • [🎶(==1==) 未添加缓存池之前](#🎶(==1==) 未添加缓存池之前)
    • [🎶(==2==) 添加缓存池后](#🎶(==2==) 添加缓存池后)
    • [🎶(==3==)缓存池 脚本](#🎶(==3==)缓存池 脚本)
    • ⭐🅰️⭐

⭐前言⭐

  • 添加缓存池的本质其实就是资源的循环利用,减少多次CG。也就是说,当我们需要销毁一个物体的时候我们需要用到的story,但是多次的destroy,它会触发我们的CG回收,那此时我们如果说用一个列表或者是字典。形成了一个缓存池,让他临时存放,我们需要多次销毁的一个物体的话,那么它就避免了多次产生C机的回收机制。此时我们可以选择让存进去的物体失活,需要的时候再激活,把它存取出来,让就可以进行一个循环的利用。

🎶(1) 未添加缓存池之前



🎶(2) 添加缓存池后



🎶(3)缓存池 脚本


cpp 复制代码
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.Events;

/// <summary>
/// 抽屉数据 
/// </summary>
public class PoolData
{
    //抽屉中 对象挂载的父节点
    public GameObject fatherObj;
    //对象的容器
    public List<GameObject> poolList;

    public PoolData(GameObject obj, GameObject poolObj)
    {
        //给我们的抽屉 创建一个父对象 并且把他作为我们pool(衣柜)对象的子物体
        fatherObj = new GameObject(obj.name);
        fatherObj.transform.parent = poolObj.transform;
        poolList = new List<GameObject>() {};
        PushObj(obj);
    }

    /// <summary>
    /// 往抽屉里面 压都东西
    /// </summary>
    /// <param name="obj"></param>
    public void PushObj(GameObject obj)
    {
        //失活 让其隐藏
        //obj.SetActive(false);  //可放在外部去激活
        //存起来
        poolList.Add(obj);
        //设置父对象
        obj.transform.parent = fatherObj.transform;
    }

    /// <summary>
    /// 从抽屉里面 取东西
    /// </summary>
    /// <returns></returns>
    public GameObject GetObj(Transform parent)  //参数为需要设置的父对象
    {
        GameObject obj = null;
        //取出第一个
        obj = poolList[0];
        //对物体的Y轴进行约束
        obj.GetComponent<Rigidbody>().constraints = RigidbodyConstraints.FreezePositionY;
        poolList.RemoveAt(0);
        //激活 让其显示
        obj.SetActive(true);
        //断开了父子关系
        obj.transform.SetParent(parent);

        return obj;
    }
}

/// <summary>
/// 缓存池模块 
/// </summary>
public class PoolManager : SingleManager <PoolManager>
{
    //缓存池容器 (衣柜)
    public Dictionary<string, PoolData> poolDic = new Dictionary<string, PoolData>();

    private GameObject poolObj;

    /// <summary>
    /// 从缓存池中拿东西
    /// </summary>
    /// <param name="名字"></param>
    /// <param parent="挂载的父对象"></param>
    /// <param position="实例化的位置"></param>
    /// <param rotation="实例化的角度"></param>
    /// <param callBack="委托"></param>
    public void GetObj(string name, Transform parent, UnityAction<GameObject> callBack) //第三个参数为挂载的父对象
    {
       
        if(poolDic.ContainsKey(name) && poolDic[name].poolList.Count > 0)
        {
           
    
            callBack(poolDic[name ].GetObj(parent));
        }
        else
        {
            //同步加载
            GameObject obj = GameObject.Instantiate(ResManager.GetInstance().Load<GameObject>("prefabs/" + name));
            obj.name = name;  
            callBack(obj);  //返回实例化的物品给委托
        
            通过异步加载资源 创建对象给外部用
            //ResManager .GetInstance().LoadAsync<GameObject>("prefabs/"+name, (o) =>
            //{
            //    o.name = name;
            //    callBack(o);
            //});

            //obj = GameObject.Instantiate(Resources.Load<GameObject>(name));
            //把对象名字改的和池子名字一样
            //obj.name = name;
        }
    }

    /// <summary>
    /// 换暂时不用的东西给池子
    /// </summary>
    public void PushObj(string name, GameObject obj)
    {
        if (poolObj == null)
            poolObj = new GameObject("Pool");

        //里面有抽屉
        if (poolDic.ContainsKey(name))
        {
            poolDic[name].PushObj(obj);
        }
        //里面没有抽屉
        else
        {          
            poolDic.Add(name, new PoolData(obj, poolObj));
        }
    }


    /// <summary>
    /// 清空缓存池的方法 
    /// 主要用在 场景切换时
    /// </summary>
    public void Clear()
    {
        poolDic.Clear();
        poolObj = null;
    }
}

⭐🅰️⭐


【Unityc#专题篇】之c#进阶篇】

【Unityc#专题篇】之c#核心篇】

【Unityc#专题篇】之c#基础篇】

【Unity-c#专题篇】之c#入门篇】

【Unityc#专题篇】---进阶章题单实践练习

【Unityc#专题篇】---基础章题单实践练习

【Unityc#专题篇】---核心章题单实践练习


你们的点赞👍 收藏⭐ 留言📝 关注✅是我持续创作,输出优质内容的最大动力!



相关推荐
菠萝咕噜肉i1 小时前
超详细:Redis分布式锁
数据库·redis·分布式·缓存·分布式锁
只因在人海中多看了你一眼5 小时前
分布式缓存 + 数据存储 + 消息队列知识体系
分布式·缓存
Dlwyz6 小时前
redis-击穿、穿透、雪崩
数据库·redis·缓存
Oak Zhang10 小时前
sharding-jdbc自定义分片算法,表对应关系存储在mysql中,缓存到redis或者本地
redis·mysql·缓存
门牙咬脆骨11 小时前
【Redis】redis缓存击穿,缓存雪崩,缓存穿透
数据库·redis·缓存
门牙咬脆骨11 小时前
【Redis】GEO数据结构
数据库·redis·缓存
Dlwyz16 小时前
问题: redis-高并发场景下如何保证缓存数据与数据库的最终一致性
数据库·redis·缓存
吴半杯18 小时前
Redis-monitor安装与配置
数据库·redis·缓存
ö Constancy19 小时前
设计LRU缓存
c++·算法·缓存
小王码农记19 小时前
vue中路由缓存
前端·vue.js·缓存·typescript·anti-design-vue