先看一个真机测试实例:
两个物体放在同一个 Group 里面
只创建 Cube 时,内存资源中只有 Cube 的材质球和贴图
同时创建了 Cube 和 Sphere,内存资源中拥有两个材质球和贴图
此时只删除 Cube,Cube 的材质球和贴图不会被释放,当然场景内存(SceneMemory)肯定是释放了的
只有把 Sphere 也释放掉才会同时将两个材质球和贴图都从资源内存中释放。
所以一个 Group 中的资源不会一起加载进资源,但是只要加载了进来资源,只有该 Group 全部引用都被删除之后,
资源才会被一起释放掉
测试代码:
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.AddressableAssets;
using UnityEngine.ResourceManagement.AsyncOperations;
public class testaa : MonoBehaviour
{
private List<GameObject> Cubelist = new List<GameObject>();
private List<GameObject> Spherelist = new List<GameObject>();
public void CreateCube()
{
Addressables.InstantiateAsync("Cube").Completed += delegate(AsyncOperationHandle<GameObject> handle)
{
Cubelist.Add(handle.Result);
};
}
public void CreateSphere()
{
Addressables.InstantiateAsync("Sphere").Completed += delegate(AsyncOperationHandle<GameObject> handle)
{
Spherelist.Add(handle.Result);
};
}
public void ClearCube()
{
for (int i = 0; i < Cubelist.Count; i++)
{
Addressables.Release(Cubelist[i]);
}
Cubelist.Clear();
}
public void ClearSphere()
{
for (int i = 0; i < Spherelist.Count; i++)
{
Addressables.Release(Spherelist[i]);
}
Spherelist.Clear();
}
}
下面是官方文档对 Addressable 内存的讲解,自己根据理解翻译了一下。
Mirroring load and unload
When working with Addressable Assets, the primary way to ensure proper memory management is to mirror your load and unload calls correctly. How you do so depends on your asset types and load methods. In all cases, however, the release method can either take the loaded asset, or an operation handle returned by the load. For example, during Scene creation (described below) the load returns a AsyncOperationHandle
镜像加载和卸载(有加载就要有卸载)
在使用 Addressable 时,正确的内存管理主要方法是成对儿地调用加载和卸载。如何这样做取决于您的资源类型和加载方法。但是,在所有情况下,release 方法接受的参数既可以是加载好的资源,也可以是加载资源时的句柄。例如,在场景创建期间 (如下所述),加载返回一个 AsyncOperationHandle
Asset loading
To load an asset, use Addressables.LoadAssetAsync or Addressables.LoadAssetsAsync.
This loads the asset into memory without instantiating it. Every time the load call executes, it adds one to the ref-count for each asset loaded. If you call LoadAssetAsync three times with the same address, you will get three different instances of an AsyncOperationHandle struct back, all referencing the same underlying operation. That operation has a ref-count of three for the corresponding asset. If the load succeeds, the resulting AsyncOperationHandle struct contains the asset in the .Result property. You can use the loaded asset to instantiate using Unity’s built-in instantiation methods, which does not increment the Addressables ref-count.
资源加载
要加载资源,请使用 Addressables.LoadAssetAsync 或 Addressables.LoadAssetsAsync。
这将在不实例化的情况下将资源加载到内存中。每次执行 load 调用时,它都会为每个加载的资源添加一个引用计数 (ref-count)。如果使用相同的地址调用 LoadAssetAsync 三次,您将获得三个不同的 AsyncOperationHandle 的回调实例,但是它们都引用相同的底层操作。该操作对应的资源的 ref-count 为 3。如果加载成功,返回的 AsyncOperationHandle 的. result 属性就是我们要的资源。您可以使用这个加载的资源通过 Unity 的内置实例化方法(指 GameObject.Instantiate) 来生成它,它不会增加 ref-count。
To unload the asset, use the Addressables.Release method, which decrements the ref-count. When a given asset’s ref-count is zero, that asset is ready to be unloaded, and decrements the ref-count of any dependencies.
Note: The asset may or may not be unloaded immediately, contingent on existing dependencies. For more information, read the section on when memory is cleared.
要卸载资源,请使用 Addressables.Release 方法,减少 ref-count。当给定资源的 ref-count 为 0 时,该资源就准备好被卸载了,并减少所有依赖项的 ref-count。
注意: 根据现有的依赖关系,资源可以立即卸载,也可以不立即卸载。有关更多信息,请阅读 “何时清除内存” 一节。
Scene loading
To load a Scene, use Addressables.LoadSceneAsync. You can use this method to load a Scene in Single mode, which closes all open Scenes, or in Additive mode (for more information, see documentation on Scene mode loading.
To unload a Scene, use Addressables.UnloadSceneAsync, or open a new Scene in Single mode. You can open a new Scene by either using the Addressables interface, or using the SceneManager.LoadScene or SceneManager.LoadSceneAsync methods. Opening a new Scene closes the current one, properly decrementing the ref-count.
场景加载
要加载一个场景,使用 Addressables.LoadSceneAsync。支持 Single mode 和 Additive mode。
要卸载一个场景,使用 Addressables.UnloadSceneAsync,或在直接使用 Single mode 打开一个新场景。这里说的打开新场景,既可以使用 Addressables 接口也可以使用 SceneManager.LoadScene 或 SceneManager.LoadSceneAsync 方法。打开一个新场景应当关闭当前场景,适当减少 ref-count。
GameObject instantiation
To load and instantiate a GameObject asset, use Addressables.InstantiateAsync. This instantiates the Prefab located by the specified location parameter. The Addressables system will load the Prefab and its dependencies, incrementing the ref-count of all associated assets.
GameObject 实例化
要加载并实例化 GameObject 资源,请使用 Addressables.InstantiateAsync。这个实例化的方法可以添加指定的参数(于 GameObject.Instantiate 类似, 看下 API 就知道了)。Addressables 系统将加载 Prefab 及其所有依赖项,并给所有这些相关资源增加的 ref-count。
Calling InstantiateAsync three times on the same address results in all dependent assets having a ref-count of three. Unlike calling LoadAssetAsync three times, however, each InstantiateAsync call returns an AsyncOperationHandle pointing to a unique operation. This is because the result of each InstantiateAsync is a unique instance. Another distinction between InstantiateAsync and other load calls is the optional trackHandle parameter. When set to false, you must keep the AsyncOperationHandle to use while releasing your instance. This is more efficient, but requires more development effort.
对同一个地址调用三次 InstantiateAsync 会导致所有相关资源的 ref-count 为 3。但是,与三次调用 LoadAssetAsync 不同的是每个 InstantiateAsync 调用都返回一个指向唯一操作的 AsyncOperationHandle。这是因为每个 InstantiateAsync 的结果都是唯一的实例。InstantiateAsync 和其他加载调用相比另一个区别在于有一个可选的 trackHandle 参数。当设置为 false 时,就必须通过 AsyncOperationHandle 的句柄来释放资源,而不能再通过它的. Result 了 (默认为 true,于本文第一段讲到的卸载参数相呼应)。
To destroy an instantiated GameObject, use Addressables.ReleaseInstance, or close the Scene that contains the instantiated object. This Scene can have been loaded (and thus closed) in Additive or Single mode. This Scene can also have been loaded using either the Addressables or SceneManagement API. As noted above, if you set trackHandle to false, you can only call Addressables.ReleaseInstance with the handle, not with the actual GameObject.
若要销毁实例化的 GameObject,请使用 Addressables.ReleaseInstance 或关闭包含实例化对象的场景。此场景应该是通过 Additive 模式或者 Single 模式已经加载好的场景(这样才能执行卸载)。这个场景既可以是使用 Addressables API 也可以是 SceneManagement API 加载好的。如上所述,如果将 trackHandle 设置为 false,则只能调用 Addressables.ReleaseInstance 来释放加载句柄,而不是使用实际的 GameObject。
Note: If you call Addressables.ReleaseInstance on an instance that was not created using the Addressables API, or was created with trackHandle==false, the system detects that and returns false to indicate that the method was unable to release the specified instance. The instance will not be destroyed in this case.
注意: 如果您调用 Addressables.ReleaseInstance 传入的实例并不是使用 Addressables API 创建的实例,或者使用 trackHandle==false 创建的实例,系统会检测到这一点并返回 false,以指示该方法无法释放指定的实例。在这种情况下,实例不会被销毁。
Addressables.InstantiateAsync has some associated overhead, so if you need to instantiate the same objects hundreds of times per frame, consider loading via the Addressables API, then instantiating through other methods. In this case, you would call Addressables.LoadAssetAsync, then save the result and call GameObject.Instantiate() for that result. This allows flexibility to call Instantiate in a synchronous way. The downside is that the Addressables system has no knowledge of how many instances you created, which can lead to memory issues if not properly managed. For example, a Prefab referencing a texture would no longer have a valid loaded texture to reference, causing rendering issues (or worse). These sorts of problems can be hard to track down as you may not immediately trigger the memory unload (see section on clearing memory, below).
Addressables.InstantiateAsync 有一些相关的开销,所以如果需要在每一帧中实例化数百次相同的对象,可以考虑通过 Addressables API 加载,然后通过其他方法实例化。在这种情况下,您将调用 Addressables.LoadAssetAsync,然后保存结果并为该结果调用 GameObject.Instantiate()。这允许以同步方式灵活地调用实例化。缺点是 Addressables 系统不知道您创建了多少实例,如果管理不当,可能会导致内存问题。例如,一个 Prefab 引用了一个加载不正确或者已经卸载的纹理,会导致渲染问题 (或更糟)。这类问题很难跟踪,因为您可能不会立即触发内存卸载 (参见下面关于清除内存的部分)。
Data loading
Interfaces that do not need their AsyncOperationHandle.Result released, will still need the operation itself to be released. Examples of these would be Addressables.LoadResourceLocationsAsync and Addressables.GetDownloadSizeAsync. They load data that you can access until the operation is released. This release should be done via Addressables.Release.
数据加载
加载的数据在卸载时不需要释放.Result,但是需要卸载这个加载资源的句柄 (就是理解为只能卸载句柄不能卸载. Result)。举个例子当使用
Addressables.LoadResourceLocationsAsync 和 Addressables.GetDownloadSizeAsync 加载到的数据,在其句柄被释放前都可以被访问到。释放需要通过 Addressables.Release 方法
Background interactions
Operations that do not return anything in the AsyncOperationHandle.Result field have have an optional parameter to automatically release the operation handle on completion. If you have no further need for one of these operation handles after it has completed, set the autoReleaseHandle parameter to true to make sure the operation handle is cleaned up. The scenario where you would want autoReleaseHandle to be false would be if you needed to check the Status of the operation handle after it has completed. Examples of these interfaces are Addressables.DownloadDependenciesAsync and Addressables.UnloadScene.
Background interactions?
如果 AsyncOperationHandle.Result 中不包含任何东西,这里有一个可选参数用于在完成时自动释放操作句柄。如果在这些操作句柄完成后,您不再需要这些操作句柄,则将 autoReleaseHandle 参数设置为 true,以确保清除了操作句柄。如果需要在操作句柄完成后检查其状态,则将 autoReleaseHandle 设置为 false。比如 Addressables.DownloadDependenciesAsync 和 Addressables.UnloadScene。
The Addressables Event Viewer
Use the Addressables Event Viewer window to monitor the ref-counts of all Addressables system operations. To access the window in the Editor, select Window > Asset Management > Addressables > Event Viewer.
Important: In order to view data in the Event Viewer, you must enable the Send Profiler Events setting in your AddressableAssetSettings object’s Inspector.
Addressables 事件查看器
使用 Addressables 事件查看器窗口监视所有 Addressables 系统操作的 ref-counts。要访问编辑器中的窗口,请 select Window > Asset Management > Addressables > Event Viewer。
重要提示: 为了在事件查看器中查看数据,您必须在您的 AddressableAssetSettings 对象的检查器中启用 Send Profiler Events 设置。
The following information is available in the Event Viewer:
- A white vertical line indicates the frame in which a load request occurred.
- A blue background indicates that an asset is currently loaded.
- The green part of the graph indicates an asset’s current ref-count.
事件查看器提供以下信息:
一条白色的竖线表示每帧发生的加载请求。
蓝色背景表示当前已经加载的资源。
图中的绿色部分表示资源当前的 ref-count。
Note that the Event Viewer is only concerned with ref-counts, not memory consumption (see section on clearing memory, below, for more information).
Listed under the Assets column, you will see a row for each of the following, per frame:
- FPS: The frames per second count.
- MonoHeap: The amount of RAM in use.
- Event Counts: The total number of events in a frame.
- Asset requests: Displays the reference count on an operation over time. If the asset request has any dependencies, a triangle appears that you can click on to view the children’s request operations.
You can click the left and right arrows in order to step through the frames one by one, or click Current to jump to the latest frame. Press the + button to expand a row for more details.
请注意,事件查看器只关心 ref-counts,而不关心内存消耗 (有关清除内存的详细信息,请参阅下面的部分)。
在 “Asset” 栏下列出的每一帧中,您可以看到逐行看到每一个资源每一帧的情况:
FPS: 每秒帧数。
MonoHeap: 内存使用的总量。
Event Counts: 一帧中事件的总数。
Asset requests: 显示一个操作随时间的引用计数。如果资产请求有任何依赖项,则会出现一个三角形,您可以单击该三角形查看子请求操作。
您可以单击左箭头和右箭头逐帧观察,或者单击 Current 跳转到最新的帧。按 + 按钮展开一行以获得更多详细信息。
The information displayed in the Event Viewer is related to the build script you use to create Play mode data.
When using the Event Viewer, avoid the Use Asset Database built script because it does not account for any shared dependencies among the assets. Use the Simulate Groups script or the Use Existing Build script instead, but the latter is better suited for the Event Viewer because it gives a more accurate monitoring of the ref-counts.
事件查看器中显示的信息与在 build script 中选择的游戏模式有关 (详细信息见 build script)
使用事件查看器时,应避免使用 Use Asset Database 模式,因为它不考虑资产之间的任何依赖关系。使用 Simulate Groups 或 Use Existing Build 模式,但是后者更适合于事件查看器,因为它可以更准确地监视 ref-counts。
When is memory cleared?
An asset no longer being referenced (indicated by the end of a blue section in the profiler) does not necessarily mean that asset was unloaded. A common applicable scenario involves multiple assets in an asset bundle. For example:
- You have three assets (tree, tank, and cow) in an asset bundle (stuff).
- When tree loads, the profiler displays a single ref-count for tree, and one for stuff.
- Later, when tank loads, the profiler displays a single ref-count for both tree and tank, and two ref-counts for the stuff bundle.
- If you release tree, it’s ref-count becomes zero, and the blue bar goes away.
何时清除内存?
不再被引用的资源(profiler 面板中的蓝色部分)并不一定意味着资源产已被卸载。一个常见的应用场景涉及到一个资源包中包含多个资源。例如:
- 您有三个资源(“树”,“坦克”,“牛”)在同一个资源包 (“东西”)。
- 当 “树” 加载时,“树”的 ref-count +1,“东西”的 ref-count +1。
- 稍后,当 “坦克” 加载时,“树”和 “坦克” 的 ref-count 均为 1,并且 “东西” 包的 ref-count 为 2。
- 如果你释放 “树”,它的 ref-count 就会变成 0。
In this example, the tree asset is not actually unloaded at this point. You can load an asset bundle, or its partial contents, but you cannot partially unload an asset bundle. No asset in stuff will unload until the bundle itself is completely unloaded. The exception to this rule is the engine interface Resources.UnloadUnusedAssets. Executing this method in the above scenario will cause tree to unload. Because the Addressables system cannot be aware of these events, the profiler graph only reflects the Addressables ref-counts (not exactly what memory holds). Note that if you choose to use Resources.UnloadUnusedAssets, it is a very slow operation, and should only be called on a screen that won’t show any hitches (such as a loading screen).
在这个例子中,“树” 资源实际上并没有被卸载。您可以加载资源包或其部分内容,但不能部分卸载资源包。在包本身完全卸载之前,所有资产都不会卸载。这个规则的例外是 engine 接口 Resources.UnloadUnusedAssets。在上述场景中执行此方法将导致树卸载。因为 Addressables 系统不能识别这些事件,所以 profiler 只反映 Addressables 的 ref-counts (不完全反映内存中存在的内容)。注意,如果您选择使用 Resources.UnloadUnusedAssets,这是一个非常慢的操作,应该只在一个不会显示任何游戏内容的屏幕调用 (比如加载屏幕)。