UNITY性能优化☀️一、GC介绍与Unity内存管理方法

简介: UNITY性能优化☀️一、GC介绍与Unity内存管理方法


🟥 GC是Unity管理内存的一个方法

游戏运行时使用内存来存储数据,当这些数据不再被使用时,存储这些数据的内存被释放以便于之后这些内存可以被复用。

垃圾(Garbage )是存储无用数据的内存的术语,GC(Garbage Collection 垃圾回收)是使这些内存可以再次使用的过程。

GC是Unity对内存管理的方式之一,我们的游戏可能因为GC负担过重而表现不佳,比如卡顿、掉帧。所以GC是引起性能问题的一个常见原因。

在这篇文章中,我们将介绍GC的工作原理,和在什么情况下会触发GC、如何减少GC对游戏的影响。



🟧 频繁GC会造成帧率过低

GC引起的性能问题可表现为帧率过低,帧率剧烈波动或者间歇性卡顿。

但是其他问题也可能引起类似的症状。

如果你的游戏有这些性能问题,首先需要使用Unity的Profiler工具来确定这些问题是由GC引起的。

如何使用Profiler工具来确定引起性能问题的原因,可以查看 这篇教程:传送门



🟨 GC回收的原理

1️⃣ GC是个费时的操作

当堆变量超出作用域后,存储该变量的内存并没有被立即释放。无用的堆内存只在执行GC时被释放。



每次执行GC时, 将执行以下步骤:

  • 垃圾收集器检索堆上的每个对象。
  • 垃圾收集器搜索所有当前对象引用以确定堆上的对象是否仍在作用域内。
    不在作用域内的对象被标记为删除。
  • 删除被标记的对象并将内存返回给堆。

GC是个费时的操作,堆上的对象越多,代码中的引用数越多,GC就越费时。


2️⃣ 可能触发GC的三种情况

  • 堆分配时堆上的可用内存不足时触发GC。
  • GC会不时的自动运行(频率因平台而异)。
  • 手动强制调用GC

GC可能被频繁触发。每当无法从可用堆内存中实现堆分配时,就会触发GC,这意味着频繁的堆分配和释放可能导致GC频繁。


3️⃣ GC导致的问题

现在我们了解了GC在Unity内存管理中的作用,我们可以考虑可能发生的问题类型。


🚨GC可能花费相当长的时间来运行

这是最明显的问题。如果堆上有很多对象和大量的对象引用要检查,则检查所有这些对象的过程可能很慢。 这可能会导致我们的游戏卡顿或运行缓慢。


🚨GC可能在不合时宜的时刻被触发

如果CPU在我们游戏的性能关键部分已经满负荷了,那此时即使是少量的GC额外开销也可能导致我们的帧速率下降和性能问题。


🚨堆碎片

当从堆中分配内存时,会根据必须存储的数据大小从不同大小的块中的可用空间中获取内存。当这些内存块返回到堆时,堆可能分成很多由分配块分隔的小空闲块。这意味着虽然可用内存总量可能很高,但由于碎片化太过严重而无法分配一块连续的大内存块。导致GC被触发或不得不扩大堆大小。


堆内存碎片化有两个后果:

  • 一是游戏内存大小会远高于实际所需要的大小
  • 二是GC会被更频繁的触发。


🟩 Unity是怎样进行的内存管理

首先,我们要知道,Unity在运行自己的核心引擎代码,和运行我们写的代码时,内存管理使用了不同的方法。

  • 当Unity在运行自己的引擎代码时,使用手动内存管理。手动内存管理不使用GC,本文不做介绍。
  • 当Unity运行我们写的脚本时,使用自动内存管理,Unity会自动帮我们完成GC工作。


基本上来说,Unity自动内存管理这样工作:

Unity可以访问两个内存池:栈和堆(也称为托管堆)。栈用于短期存储小块数据,堆用于长期存储和较大数据段。

当创建变量时,Unity从栈或堆中申请内存,只要变量在作用域内,分配给它的内存就会一直在使用, 我们称这部分内存已被分配。

我们将栈中的变量称为栈对象,将堆中的变量称为堆对象。

当变量超出作用域时,该内存会不会再被使用,并且可以归还给原来的内存池。当内存归还给原有的内存池时,我们称该内存被释放。

栈内存在变量超出作用域时被实时释放,而堆内存在变量超出作用域之后并没有被释放并保持被分配的状态。

垃圾收集器(garbage collector)识别和释放未使用的堆内存。 垃圾收集器定期运行以清理堆。


🟦 栈和堆在内存管理上的不同

1️⃣ 在栈分配和释放时发生了什么

栈分配和释放简单快速。这是因为栈只用于在短时间内存储小数据。 分配和释放总是以可预测的顺序发生,并且具有可预测的大小。

栈的工作方式类似于栈数据类型, 它是一个简单的元素集合。

这种情况下的内存块,只能以严格的顺序添加和删除元素。 这种简单性和严格性使得它变得非常快速。

当一个变量存储在栈上时,它的内存就是简单地从栈顶分配。 栈变量超出作用域时,用于存储该变量的内存将立即返回栈进行重用。


2️⃣ 在堆分配时发生了什么

堆分配比栈分配复杂的多。因为:

  • 堆可以用来存储长期和短期数据
  • 堆可以存储不同类型、不同大小的数据
  • 堆分配和释放也并不总是按可预测的顺序进行
  • 堆且可能需要大小差距巨大的内存块

当一个堆变量创建时,将执行以下步骤:

1、检查堆上是否有足够的空闲内存

如果有,则分配该变量的内存。如果没有,Unity触发GC试图释放未使用的堆内存。

这个操作可能很慢。如果GC之后堆内存足够,则该变量的内存被分配。


2、如果GC之后堆上还是没有足够的空闲内存

Unity将向操作系统申请更多内存以扩大堆大小。这个操作可能很慢。

之后该变量的内存被分配。

堆分配可能会很慢,特别在必须执行GC和扩大堆大小时。

相关文章
|
17天前
|
存储 安全 iOS开发
内存卡怎么格式化?6个格式化方法供你选
随着使用时间的增加,内存卡可能会因为数据积累、兼容性或是文件系统损坏等原因需要进行格式化。那么怎样正确格式化内存卡呢?格式化内存卡的时候需要注意什么呢?本文会给大家提供详细的步骤,帮助大家轻松完成格式化内存卡的操作。
|
2月前
|
监控 JavaScript Java
Node.js中内存泄漏的检测方法
检测内存泄漏需要综合运用多种方法,并结合实际的应用场景和代码特点进行分析。及时发现和解决内存泄漏问题,可以提高应用的稳定性和性能,避免潜在的风险和故障。同时,不断学习和掌握内存管理的知识,也是有效预防内存泄漏的重要途径。
195 52
|
2月前
|
传感器 人工智能 物联网
C 语言在计算机科学中尤其在硬件交互方面占据重要地位。本文探讨了 C 语言与硬件交互的主要方法,包括直接访问硬件寄存器、中断处理、I/O 端口操作、内存映射 I/O 和设备驱动程序开发
C 语言在计算机科学中尤其在硬件交互方面占据重要地位。本文探讨了 C 语言与硬件交互的主要方法,包括直接访问硬件寄存器、中断处理、I/O 端口操作、内存映射 I/O 和设备驱动程序开发,以及面临的挑战和未来趋势,旨在帮助读者深入了解并掌握这些关键技术。
72 6
|
5月前
|
语音技术 开发工具 图形学
Unity与IOS⭐一、百度语音IOS版Demo调试方法
Unity与IOS⭐一、百度语音IOS版Demo调试方法
|
3月前
|
缓存 算法 Java
JVM知识体系学习六:JVM垃圾是什么、GC常用垃圾清除算法、堆内存逻辑分区、栈上分配、对象何时进入老年代、有关老年代新生代的两个问题、常见的垃圾回收器、CMS
这篇文章详细介绍了Java虚拟机(JVM)中的垃圾回收机制,包括垃圾的定义、垃圾回收算法、堆内存的逻辑分区、对象的内存分配和回收过程,以及不同垃圾回收器的工作原理和参数设置。
140 4
JVM知识体系学习六:JVM垃圾是什么、GC常用垃圾清除算法、堆内存逻辑分区、栈上分配、对象何时进入老年代、有关老年代新生代的两个问题、常见的垃圾回收器、CMS
|
2月前
|
Android开发 开发者
Android性能优化——内存管理的艺术
Android性能优化——内存管理的艺术
|
2月前
|
存储 缓存 监控
|
3月前
|
机器学习/深度学习 算法 物联网
大模型进阶微调篇(一):以定制化3B模型为例,各种微调方法对比-选LoRA还是PPO,所需显存内存资源为多少?
本文介绍了两种大模型微调方法——LoRA(低秩适应)和PPO(近端策略优化)。LoRA通过引入低秩矩阵微调部分权重,适合资源受限环境,具有资源节省和训练速度快的优势,适用于监督学习和简单交互场景。PPO基于策略优化,适合需要用户交互反馈的场景,能够适应复杂反馈并动态调整策略,适用于强化学习和复杂用户交互。文章还对比了两者的资源消耗和适用数据规模,帮助读者根据具体需求选择最合适的微调策略。
1025 5
|
3月前
|
缓存 监控 Java
在使用 Glide 加载 Gif 动画时避免内存泄漏的方法
【10月更文挑战第20天】在使用 Glide 加载 Gif 动画时,避免内存泄漏是非常重要的。通过及时取消加载请求、正确处理生命周期、使用弱引用、清理缓存和避免重复加载等方法,可以有效地避免内存泄漏问题。同时,定期进行监控和检测,确保应用的性能和稳定性。需要在实际开发中不断积累经验,根据具体情况灵活运用这些方法,以保障应用的良好运行。
|
3月前
|
存储 算法 Java
Java虚拟机(JVM)的内存管理与性能优化
本文深入探讨了Java虚拟机(JVM)的内存管理机制,包括堆、栈、方法区等关键区域的功能与作用。通过分析垃圾回收算法和调优策略,旨在帮助开发者理解如何有效提升Java应用的性能。文章采用通俗易懂的语言,结合具体实例,使读者能够轻松掌握复杂的内存管理概念,并应用于实际开发中。

热门文章

最新文章