news 2026/6/21 19:08:50

MyFramework:ResourceRef 资源引用凭证设计

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
MyFramework:ResourceRef 资源引用凭证设计

ResourceManager加载资源时,没有直接把UnityEngine.Object返回给业务层,而是返回一个ResourceRef<T>

这个类很小,但它承担了资源引用生命周期管理的核心逻辑。


一、代码

ResourceRef<T>的实现如下:

public class ResourceRef<T> : ClassObject where T : UObject { protected T mResource; // 引用的资源 protected long mToken; // 引用凭证,一般不允许外部直接访问 public override void resetProperty() { base.resetProperty(); mResource = null; mToken = 0; } public void setResource(T res) { mResource = res; if (mResource == null) { logError("resource is null"); return; } mToken = mResourceManager.addReference(mResource); } public bool isValid() { return mResource != null; } public T getResource() { return mResource; } public long getToken() { return mToken; } // 在UN_CLASS时自动被调用 public override void destroy() { base.destroy(); if (mResource == null) { logError("resource is null"); return; } mResourceManager.removeReference(mResource, ref mToken); } // 对当前资源新创建一个引用对象出来,用于使多个地方对同一个资源拥有生命周期所有权 public ResourceRef<T> copyRef() { CLASS(out ResourceRef<T> newObjRef).setResource(mResource); return newObjRef; } }

业务层拿到的不是裸资源,而是:

ResourceRef<Texture> refTex;

真正使用时再取资源:

Texture tex = refTex.getResource();

释放时也不是直接卸载资源,而是释放引用对象:

mResourceManager.unload(ref refTex);

二、加载时加引用

同步加载资源时,ResourceManager会先加载出真实资源:

T res = mAssetBundleLoader.loadAsset<T>(name);

资源不为空时,再创建ResourceRef<T>

CLASS(out ResourceRef<T> resRef).setResource(res); return resRef;

setResource()里会调用:

mToken = mResourceManager.addReference(mResource);

也就是说,只要业务层拿到一个ResourceRef<T>,资源系统内部就会增加一份引用凭证。


三、token 不是简单计数

ResourceManager中不是只保存一个整数引用计数,而是保存 token 集合:

protected Dictionary<int, HashSet<long>> mReferenceTokenList = new(); protected Dictionary<int, UObject> mInstanceIDToUObject = new();

增加引用时:

public long addReference(UObject res) { long token = ++mTokenSeed; int instanceID = res.GetInstanceID(); mInstanceIDToUObject.TryAdd(instanceID, res); if (!mReferenceTokenList.getOrAddNew(instanceID).Add(token)) { logError("添加资源引用凭证失败:" + token); } return token; }

移除引用时:

public void removeReference(UObject res, ref long token) { if (!mReferenceTokenList.TryGetValue(res.GetInstanceID(), out var list) || !list.Remove(token)) { logError("移除资源引用凭证失败,可能是重复移除一个资源:" + token); } token = 0; }

这里的关键是:

每个 ResourceRef 都有自己的 token 同一个资源可以有多个 token 释放时只移除当前 ResourceRef 的 token token 清空后可以检测重复释放

如果只用一个整数引用计数,重复释放很难定位。

使用 token 后,如果同一个ResourceRef被重复释放,第二次移除 token 就会失败,并打印错误。


四、tokenSeed 放在 ResourceManager

mTokenSeed放在ResourceManager中:

protected static long mTokenSeed;

代码注释里写得很清楚:

不能放在 ResourceRef<T> 中, 因为每个模板类型都有一个静态变量, 这样就不能保证同一个资源的引用凭证在不同模板类型中是唯一的。

这是一个容易忽略的细节。

如果写成:

public class ResourceRef<T> { private static long mTokenSeed; }

那么这些类型会各自有一份静态变量:

ResourceRef<Texture>.mTokenSeed ResourceRef<Sprite>.mTokenSeed ResourceRef<UObject>.mTokenSeed

同一个资源可能通过不同泛型类型包装。

如果 token 分散在不同泛型类里生成,就可能出现重复 token。

所以 token 生成必须放在统一的ResourceManager中。


五、为什么用 InstanceID

引用表的 Key 使用的是:

res.GetInstanceID()

不是直接用UnityEngine.Object

代码注释说明了原因:

UObject 重载了 ==, 外部卸载 UObject 后可能出现 GetHashCode 不变, 但引用资源为空的问题, 所以使用 GetInstanceID 作为 Key。

Unity 的Object和普通 C# 对象不完全一样。

它有自己的生命周期。

资源被 Unity 销毁后,C# 引用还可能存在,但== null的行为已经被 Unity 重载。

如果直接拿UObject当 Dictionary Key,后续判断会变得不稳定。

GetInstanceID()做索引,逻辑更明确。


六、释放时不立刻卸载

释放ResourceRef<T>时,只是移除 token。

mResourceManager.removeReference(mResource, ref mToken);

真正卸载资源不是在这里立即完成。

ResourceManager.update()会定时检查引用表:

protected const float CHECK_REF_INTERVAL = 3.0f;

检查逻辑是:

foreach (var item in mReferenceTokenList) { if (item.Value.isEmpty()) { if (willRemoveList == null) { LIST(out willRemoveList); } willRemoveList.add(item.Key); } }

发现某个资源的 token 集合为空后,再统一卸载:

foreach (int id in willRemoveList) { mInstanceIDToUObject.Remove(id, out UObject item); mReferenceTokenList.Remove(id); unloadInternal(item); }

这样做有两个好处:

资源释放和真实卸载解耦 避免同一帧频繁加载和卸载

业务层只负责释放引用。

资源系统决定什么时候真正卸载资源。


七、copyRef 的意义

ResourceRef<T>提供了:

public ResourceRef<T> copyRef() { CLASS(out ResourceRef<T> newObjRef).setResource(mResource); return newObjRef; }

它不是简单复制对象引用。

它会创建一个新的ResourceRef<T>,并重新调用setResource()

这意味着:

同一个资源 新的 ResourceRef 新的 token 独立生命周期

适合这种情况:

一个资源加载后,需要交给多个模块使用 每个模块都应该独立释放自己的引用 最后一个引用释放后,资源才允许卸载

如果只是把同一个ResourceRef<T>传给多个地方,就会出现所有权不清楚的问题。

一个模块释放后,其他模块可能还在使用。

copyRef()让多个持有者拥有独立引用。


八、不是裸资源所有权

如果业务层直接拿TextureSpritePrefab,资源系统无法知道谁还在使用它。

ResourceRef<T>的作用是把资源使用权显式化。

拿到 ResourceRef 表示持有一份资源引用 释放 ResourceRef 表示归还这份资源引用

资源本体可以被多个地方共享。

引用凭证属于每个持有者。

这个设计比裸传资源更适合框架统一管理资源生命周期。


九、和对象池配合

ResourceRef<T>继承自ClassObject

它本身也是池化对象。

创建时:

CLASS(out ResourceRef<T> resRef)

释放时:

UN_CLASS(ref res);

释放过程会走:

destroy ↓ removeReference ↓ resetProperty ↓ 回收到 ClassPool

destroy()负责移除资源引用。

resetProperty()负责清空自身字段。

这和 MyFramework 的对象池规则保持一致。


十、精巧点

ResourceRef<T>精巧的地方主要有四个。

1. 引用不是 int,而是 token 集合

可以检测重复释放,也能让每个持有者有独立凭证。

2. tokenSeed 不放在泛型类中

避免不同ResourceRef<T>类型各自产生重复 token。

3. 使用 InstanceID 追踪 Unity Object

避免 UnityObject重载==后带来的 Dictionary Key 问题。

4. copyRef 创建独立引用

同一个资源可以交给多个模块使用,每个模块释放自己的引用。


总结

ResourceRef<T>的设计不是简单包一层资源对象。

它解决的是资源所有权问题。

核心流程是:

加载资源 ↓ 创建 ResourceRef ↓ ResourceManager 生成 token ↓ 业务层持有 ResourceRef ↓ 释放 ResourceRef ↓ 移除 token ↓ 所有 token 清空后资源进入卸载流程

这个设计让资源生命周期从“谁拿着裸对象”变成“谁持有引用凭证”。

在 MyFramework 这种同时支持 AssetDatabase、AssetBundle、异步加载、子资源、下载和卸载的资源系统里,这层引用凭证非常关键。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/21 18:48:04

ComfyUI Inpaint Nodes:解锁AI图像修复与扩展的5大核心功能

ComfyUI Inpaint Nodes&#xff1a;解锁AI图像修复与扩展的5大核心功能 【免费下载链接】comfyui-inpaint-nodes Nodes for better inpainting with ComfyUI: Fooocus inpaint model for SDXL, LaMa, MAT, and various other tools for pre-filling inpaint & outpaint are…

作者头像 李华
网站建设 2026/6/21 18:39:48

大语言模型实时持续遗忘技术CURaTE:原理、实现与部署指南

1. 项目概述&#xff1a;当大模型需要“选择性失忆”最近在折腾本地部署大语言模型&#xff08;LLM&#xff09;时&#xff0c;我遇到了一个挺有意思&#xff0c;也相当棘手的问题&#xff1a;怎么让一个已经训练好的大模型&#xff0c;忘掉某些特定的知识&#xff1f;这听起来…

作者头像 李华
网站建设 2026/6/21 18:39:43

Ubuntu 18.04 上构建可审计的 Zabbix 安全监控基线

1. 为什么在 Ubuntu 18.04 上部署 Zabbix 监控不是“装完就跑”&#xff0c;而是必须从安全基线开始设计Zabbix 是我过去八年里在金融、教育和中小制造企业现场部署频率最高的开源监控系统——不是因为它最炫&#xff0c;而是它在“可控性”和“可审计性”之间找到了极难复制的…

作者头像 李华
网站建设 2026/6/21 18:21:09

3分钟搞定抖音评论数据采集:TikTokCommentScraper终极实战指南

3分钟搞定抖音评论数据采集&#xff1a;TikTokCommentScraper终极实战指南 【免费下载链接】TikTokCommentScraper 项目地址: https://gitcode.com/gh_mirrors/ti/TikTokCommentScraper 你是否曾为收集抖音热门视频的用户评论而烦恼&#xff1f;TikTokCommentScraper是…

作者头像 李华
网站建设 2026/6/21 18:10:19

终极实战指南:Box64让ARM设备完美运行x86程序的完整解决方案

终极实战指南&#xff1a;Box64让ARM设备完美运行x86程序的完整解决方案 【免费下载链接】box64 Box64 - Linux Userspace x86_64 Emulator with a twist, targeted at ARM64, RV64 and LoongArch Linux devices 项目地址: https://gitcode.com/gh_mirrors/bo/box64 Box…

作者头像 李华
网站建设 2026/6/21 18:08:52

终极Alienware控制指南:如何用500KB工具替代臃肿的AWCC软件

终极Alienware控制指南&#xff1a;如何用500KB工具替代臃肿的AWCC软件 【免费下载链接】alienfx-tools Alienware systems lights, fans, and power control tools and apps 项目地址: https://gitcode.com/gh_mirrors/al/alienfx-tools 你是否厌倦了Alienware Command…

作者头像 李华