# 优化Unity WebGL的内存 ## 一、内存与OOM Unity WebGL游戏通常比普通H5(JS)游戏占用更大的内存,在操作系统的控制策略下超出阈值时非常容易被OOM。 为了提高游戏在中低端机型的稳定性,内存优化极为重要。那么,多大的内存是合理的呢? 我们建议内存峰值控制在以下范围: 1. Android: 低档机 < 1.2G, 中高档机 < 1.5G 2. iOS: 低档机 < 1G, 中高档机 < 1.4G 相对而言,Android机型的内存更为宽松。 iOS低档机为iPhone 6sp/7/8等2G机型为主,中高档机为iPhone 7P/8P/iPhoneX/iPhone11等。 ## 二、Unity WebGL适配小游戏的内存结构 Unity WebGL内存结构可先参考: [Unity博客:了解 Unity WebGL 中的内存 (Understanding Memory in Unity WebGL)](https://blogs.unity3d.com/2016/09/20/understanding-memory-in-unity-webgl/?_ga=2.221731928.1756133398.1638172136-1917727044.1634213664) [Unity博客:Unity WebGL 内存:Unity 堆 (Unity WebGL Memory: The Unity Heap)](https://blog.unity.com/technology/unity-webgl-memory-the-unity-heap) Unity WebGL是以WebAssembly+WebGL技术为基础的应用,运行在浏览器环境,因此游戏内存的分配也是完全托管在这个环境中。 适配在小游戏后,小游戏进程也就成为了容器,虽然不再是标准的浏览器,但内存组成结构与上图基本一致。以下是小游戏环境的注意事项: - DOM:浏览器页面元素,Cavas等。在小游戏环境中并不存在DOM,但依然会存在一些基本消耗,比如小游戏公共库,Canvas画布等。典型地,***小游戏公共库约占用内存100~150MB,Canvas 画布与设备物理分辨率相关***,比如iPhone 11 Promax占用约80MB。 - Unity Heap: 用于存储所有状态、托管的对象和本机对象以及当前加载的资源和场景的内存。举例,游戏逻辑分配的C#对象等托管内存,以及Unity管理的AssetBundle对象、场景结构等本机内存。具体占用通过WX.LogUnityHeapMem打印到控制台。 - Asset Data: 原生APP通常不会有这类内存,AssetData需要文件的同步读写能力,但浏览器环境并不支持。Emscripten使用[文件系统](https://emscripten.org/docs/api_reference/Filesystem-API.html)模拟Linux/POSIX接口,***代价是占用与文件同等大小的内存***。 - AssetBundles: 此处指的是下载时产生WebHttpRequest的临时内存,并非AB在Native内存的占用。***当包体较大或网络并发较大时,下载容易产生内存峰值,虽然为临时内存缺很可能突破OOM阈值导致Crash***。 - WebAudio:Unity将音频传递给容器(浏览器或小游戏)后,播放音频时将占用的内存。目前自带音频系统存在较大内存和性能问题,Unity2021之前不支持内存压缩音频,因此音频加载后将被完整解压,***非压缩音频会导致极大内存占用***。 - Memory File System: 由于Emscripten的[文件系统](https://emscripten.org/docs/api_reference/Filesystem-API.html)。除了前面提到的Asset Data外,任何需要File.Read/Write同步操作的数据都将占用内存,***通常包括:首资源包、Addressable Cache机制、WWW.LoadFromCacheOrDownload等Cache API。开发者不应使用此类接口。*** - Comiled Code: WebAssembly需要编译,在Android v8、iOS JavascriptCore中还需要大量内存进行JIT优化,***开发者应减少非必要第三方插件,使用代码分包能力进行内存控制***。 ## 三、内存查看工具 开发者可由系统进程、UnityHeap、JavaScript Heap等方面去分析和查看游戏内存。 ### 3.1 进程总内存 查看总内存时,我们需要先确定监控的小游戏进程名称: - Android:WeChat AppBrand1/2 - iOS:普通模式WeChat、高性能模式(WebContent) #### Instruments in Xcode(iOS) 使用Activity Monitor,选择对应的设备-all processes-捕捉,即可看到所有进程的CPU与内存情况. #### Perfdog(Android or iOS) 使用[Perfdog](https://perfdog.qq.com)选择对应的设置-进程名,即可看到相关性能数据,iOS设备应以紫色的XcodeMemory为准。 ### 3.2 UnityHeap 开发体、体验版小游戏默认会在左上角呈现PerformenceStats, 开发者可通过改面板了解性能数值. 每项指标有三个数值:当前帧、最小值、最大值 - MonoUsedSize: 托管堆(如C#业务逻辑)当前的内存使用量 - MonoHeap:托管堆的内存使用峰值 - Graphic:GPU显存使用量(部分Unity版本存在BUG) - TotalReserverdMemory:本机堆(Native)内存分配峰值 - TotalUnusedReserverdMemory:本机堆(Native)空闲内存值 - TotalAllocatedMemory:本机堆(Native)当前的内存使用量 - TotalMemorySize: UnityHeap总预分配内存大小 - DynamicMemory:UnityHeap当前使用量 开发者尤其需要关注MonoHeap、TotalReserverdMemory、DynamicMemory这三项数值。如果游戏中有C原生代码(如lua)分配的内存,则需单独统计,该部分内存包括在DynamicMemory但并不包含在托管堆和本机堆。 ### 3.2 JavaScript Heap 由于Unity WebGL是托管在浏览器环境中,因此JavaScript Heap包含了大部分(并非全部)我们关注的内存, 通常我们可以使用浏览器自带的内存工具。 #### FireFox Memory(PC) #### iOS Safari Timeline(PC or iOS) ### 四、内存优化方案 以iOS为例,一款代码(导出目录/webgl/Build/xxx.code.unityweb或code.wasm)大小为30MB的游戏占用内存为: 小游戏基础库(130MB) + Cavnas(70MB) + 编译内存(300MB) + UnityHeap(托管内存 + Native内存 + C原生代码内存) + Gfx显存 + 音频 + JavaScript内存(通常<100MB)。 假如游戏需要支持低档机型,将内存控制到1G以内,业务侧(Unity Heap, Gfx显存,音频,胶水层)需控制在500MB左右。我们此处给出转换游戏中最容易遇到的内存问题与解决方案,如果开发者遇到内存问题时请逐个排查优化。 ### 4.1 WebAssembly编译代码内存 - 问题原因:Unity WebGL将所有代码(引擎、业务代码、第三方插件)编译为跨平台的WebAssembly二进制代码,运行时需进行编译执行。编译所占用内存占用非常大(如在iOS系统,30MB未压缩代码需300MB运行时编译内存)。 - 解决办法:[使用代码分包工具](WasmSplit.md)能降低原编译代码内存50%以上。 ### 4.2 GPU纹理内存 - 问题原因:Unity 2021才开始支持移动平台的压缩纹理,使用RGBA、DXT等纹理格式将导致巨大的内存开销与运行时解压消耗。 - 解决办法:[压缩纹理优化](CompressedTexture.md)能最大程度地减少内存与解压开销。 ### 4.3 Unity Heap - 问题原因:Unity Heap是用于存储所有状态、托管的对象和本机对象,往往由于场景过大或由于业务原因造成瞬间内存峰值。***由于Unity WebGL在单首帧内无法GC***,单帧内瞬间的内存使用非常容易造成crash。同时,***Heap是只增不减且存在内存碎片的。*** - 解决办法: 1. 避免场景过大导致瞬间峰值 2. 避免过大的AssetBUndle导致瞬间峰值 3. 避免单帧内分配过多的对象 4. 建议200MB左右为宜,***不要超过300MB,切忌产生跳跃峰值***, 5. 重度游戏建议使用WX.LogUnityHeapMem查看Dynamic峰值,填写MiniGameConfig.asset中的内存为该值。 ### 4.4 首资源包与AssetBundle内存 - 问题原因:首资源包永远占用内存且无法释放;首资源包和AssetBundle自带的cache机制都会使用Emscripten使用[文件系统](https://emscripten.org/docs/api_reference/Filesystem-API.html),应避免使用。 - 解决办法:1. 减少首资源包大小,此部分始终占用内存无法释放, 使用AssetBundle; 2. AssetBundle按需加载,及时释放以节省内存; 3. AssetBundle使用时被解压占用Unity Native内存,应减少AssetBundle大小; 4. 避免使用Unity自带的文件缓存机制, ***首资源包和AssetBundle都不应使用文件Cache***; ### 4.5 音频内存 - 问题原因:Unity2021之前不支持内存压缩音频,因此音频加载后将被完整解压,***非压缩音频会导致极大内存占用***。 - 解决办法:使用小游戏SDK[音频适配优化](AudioOptimization.md)。 ### 4.6 其他常见优化手段 - [Unity加载和内存管理](https://zentia.github.io/2018/04/11/AssetBundle/) - [性能优化,进无止境-内存篇(上)](https://blog.uwa4d.com/archives/optimzation_memory_1.html) - [性能优化,进无止境-内存篇(下)](https://blog.uwa4d.com/archives/optimzation_memory_2.html) - [全面理解Unity加载和内存管理](https://gameinstitute.qq.com/community/detail/100741) ## 五、QA 1. 在Unity Profiler看到内存才200MB+,是否代表游戏内存无问题 不是。游戏占用内存必须以真机环境为准,使用Perfdog(Android or iOS)或 Instruments in Xcode(iOS)测试对应进程的内存占用。Unity Profiler仅能看到Unity Heap相关内存,并不包含小游戏公共库、Cavas、WebAssembly编译以及容器其他内存。