Unity GC垃圾回收

今天记录一些GC相关的知识点,加上一些自己的理解。

英文原文:https://unity3d.com/de/learn/tutorials/topics/performance-optimization/optimizing-garbage-collection-unity-games?playlist=44069

翻译参考:http://www.cnblogs.com/zblade/p/6445578.html

 

GC (garbage collection)简介

在游戏运行的时候,数据主要存储在内存中,当游戏的数据在不需要的时候,存储当前数据的内存就可以被回收以再次使用。内存垃圾是指当前废弃数据所占用的内存,垃圾回收(GC)是指将废弃的内存重新回收再次使用的过程。

Unity中将垃圾回收当作内存管理的一部分,如果游戏中废弃数据占用内存较大,则游戏的性能会受到极大影响,此时垃圾回收会成为游戏性能的一大障碍点。

 

Unity内存管理机制简介

Unity主要采用自动内存管理的机制,开发时在代码中不需要详细地告诉unity如何进行内存管理,unity内部自身会进行内存管理。这和使用C++开发需要随时管理内存相比,有一定的优势,当然带来的劣势就是需要随时关注内存的增长。

unity的自动内存管理可以理解为以下几个部分:

1.unity内部有两个内存管理池:堆内存和堆栈内存。堆栈内存(stack)主要用来存储较小的和短暂的数据,堆内存(heap)主要用来存储较大的和存储时间较长的数据。

2.unity中的变量只会在堆栈或者堆内存上进行内存分配,值类型变量都在堆栈上进行内存分配,其他类型的变量都在堆内存上分配。

3.只要变量处于激活状态,则其占用的内存会被标记为使用状态,则该部分的内存处于被分配的状态。

4.一旦变量不再激活,则其所占用的内存不再需要,该部分内存可以被回收到内存池中被再次使用,这样的操作就是内存回收。处于堆栈上的内存回收及其快速,处于堆上的内存并不是及时回收的,此时其对应的内存依然会被标记为使用状态。

5.垃圾回收主要是指堆上的内存分配和回收,unity中会定时对堆内存进行GC操作。

 

堆栈内存分配和回收机制

堆栈上的内存分配和回收十分快捷简单,因为堆栈上只会存储短暂的或者较小的变量。内存分配和回收都会以一种顺序和大小可控制的形式进行。

堆栈的运行方式就像stack: 其本质只是一个数据的集合,数据的进出都以一种固定的方式运行。正是这种简洁性和固定性使得堆栈的操作十分快捷。当数据被存储在堆栈上的时候,只需要简单地在其后进行扩展。当数据失效的时候,只需要将其从堆栈上移除。

堆内存分配和回收机制

堆内存上的内存分配和存储相对而言更加复杂,主要是堆内存上可以存储短期较小的数据,也可以存储各种类型和大小的数据。其上的内存分配和回收顺序并不可控,可能会要求分配不同大小的内存单元来存储数据。

堆上的变量在存储的时候,

1.首先unity会先检测是否有足够的闲置内存单元用来存储数据,如果有,则分配对应大小的内存单元;

2.如果没有,就触发垃圾回收(GC)来释放不再被使用的堆内存(缓慢的操作),如果垃圾回收后有足够大小的内存单元,则进行内存分配。

3.如果还不够,则会扩展堆内存的大小(缓慢的操作),最后分配对应大小的内存单元给变量。

堆内存的分配有可能会变得十分缓慢,特别是在需要垃圾回收和堆内存需要扩展的情况下,通常需要减少这样的操作次数。

 

GC相关的一些信息

GC的操作过程:

当堆内存上一个变量不再处于激活状态的时候,其所占用的内存并不会立刻被回收,不再使用的内存只会在GC的时候才会被回收。其操作如下

1.GC会检查堆内存上的每个存储变量;

2.对每个变量会检测其引用是否处于激活状态;

3.如果变量的引用不再处于激活状态,则会被标记为可回收;

4.被标记的变量会被移除,其所占有的内存会被回收到堆内存上。

GC操作是一个极其耗费的操作,堆内存上的变量或者引用越多则其运行的操作会更多,耗费的时间越长。

 

何时触发GC:

主要有三个操作会触发垃圾回收:

1.在堆内存上进行内存分配操作而内存不够的时候都会触发垃圾回收来利用闲置的内存;

2.GC会自动的触发,不同平台运行频率不一样;

3.GC可以被强制执行。

特别是在堆内存上进行内存分配时内存单元不足够的时候,GC会被频繁触发,这就意味着频繁在堆内存上进行内存分配和回收会触发频繁的GC操作。

 

GC操作带来的问题:

1.需要大量的时间来运行,可能会使得游戏运行缓慢。其次GC可能会在关键时候运行,例如在CPU处于游戏的性能运行关键时刻,此时任何一个额外的操作都可能会带来极大的影响,使得游戏帧率下降。

2.堆内存的碎片划。当一个内存单元从堆内存上分配出来,其大小取决于其存储的变量的大小。当该内存被回收到堆内存上的时候,有可能使得堆内存被分割成碎片化的单元。也就是说堆内存总体可以使用的内存单元较大,但是单独的内存单元较小,在下次内存分配的时候不能找到合适大小的存储单元,这也会触发GC操作或者堆内存扩展操作。

堆内存碎片会造成两个结果,一个是游戏占用的内存会越来越大,一个是GC会更加频繁地被触发。

 

利用profiler window 来检测堆内存分配(unity工具栏Window->Profiler打开):

在CPU usage分析窗口中,我们可以检测任何一帧cpu的内存分配情况。其中一个列是GC Alloc,通过分析其来定位是什么函数造成大量的堆内存分配操作。一旦定位该函数,我们就可以分析解决其造成问题的原因从而减少内存垃圾的产生。现在Unity5.5的版本,还提供了deep profiler的方式深度分析GC垃圾的产生。如图

Unity GC垃圾回收_第1张图片

 

优化方案

 降低GC影响的方法

1.减少GC的运行次数;

2.减少单次GC的运行时间;

3.将GC的运行时间延迟,避免在关键时候触发,比如可以在场景加载的时候调用GC

主要策略为

1.对游戏进行重构,减少堆内存的分配和引用的分配。更少的变量和引用会减少GC操作中的检测个数从而提高GC的运行效率。

2.降低堆内存分配和回收的频率,尤其是在关键时刻。也就是说更少的事件触发GC操作,同时也降低堆内存的碎片化。

3.我们可以试着测量GC和堆内存扩展的时间,使其按照可预测的顺序执行。当然这样操作的难度极大,但是这会大大降低GC的影响。

 

具体如下:

减少内存垃圾的数量

1.缓存

如果在代码中反复调用某些造成堆内存分配的函数但是其返回结果并没有使用,这就会造成不必要的内存垃圾,我们可以缓存这些变量来重复利用,这就是缓存。

例如下面的代码每次调用的时候就会造成堆内存分配,主要是每次都会分配一个新的数组:

void OnTriggerEnter(Collider other) {
     Renderer[] allRenderers = FindObjectsOfType();
     ExampleFunction(allRenderers);      
}

对比下面的代码,只会生产一个数组用来缓存数据,实现反复利用而不需要造成更多的内存垃圾:

Renderer[] allRenderers;
 
void Start() {
   allRenderers = FindObjectsOfType();
}
 
void OnTriggerEnter(Collider other) {
    ExampleFunction(allRenderers);
}

2.不要在频繁调用的函数中反复进行堆内存分配

在MonoBehaviour中,如果我们需要进行堆内存分配,最坏的情况就是在其反复调用的函数中进行堆内存分配,例如Update()和LateUpdate()函数这种每帧都调用的函数,这会造成大量的内存垃圾。我们可以考虑在Start()或者Awake()函数中进行内存分配,这样可以减少内存垃圾。

下面的例子中,update函数会多次触发内存垃圾的产生:

void Update() {
    ExampleGarbageGenerationFunction(transform.position.x);
}

通过一个简单的改变,我们可以确保每次在x改变的时候才触发函数调用,这样避免每帧都进行堆内存分配:

float previousTransformPositionX;

void Update() {
    float transformPositionX = transform.position.x;
    if(transfromPositionX != previousTransformPositionX) {
        ExampleGarbageGenerationFunction(transformPositionX);    
        previousTransformPositionX = trasnformPositionX;
    }
}

另外的一种方法是在update中采用计时器,特别是在运行有规律但是不需要每帧都运行的代码中,例如:

float timeSinceLastCalled;
float delay = 1f;
void Update() {
    timSinceLastCalled += Time.deltaTime;
    if(timeSinceLastCalled > delay) {
         ExampleGarbageGenerationFunction();
         timeSinceLastCalled = 0f;
    }
}

3.清除链表

在堆内存上进行链表的分配的时候,如果该链表需要多次反复的分配,我们可以采用链表的clear函数来清空链表从而替代反复多次的创建分配链表。

void Update() {
    List myList = new List();
    PopulateList(myList);       
}

通过改进,我们可以将该链表只在第一次创建或者该链表必须重新设置的时候才进行堆内存分配,从而大大减少内存垃圾的产生:

List myList = new List();
void Update() {
    myList.Clear();
    PopulateList(myList);
}

4.对象池

即便我们在代码中尽可能地减少堆内存的分配行为,但是如果游戏有大量的对象需要产生和销毁依然会造成GC。对象池技术可以通过重复使用对象来降低堆内存的分配和回收频率。对象池在游戏中广泛的使用,特别是在游戏中需要频繁的创建和销毁相同的游戏对象的时候,例如枪的子弹这种会频繁生成和销毁的对象。

可以看之前写的:Unity 对象池

 

造成不必要的堆内存分配的因素

1.字符串

在c#中,字符串是引用类型变量而不是值类型变量,即使看起来它是存储字符串的值的。这就意味着字符串会造成一定的内存垃圾,由于代码中经常使用字符串,所以我们需要对其格外小心。

c#中的字符串是不可变更的,也就是说其内部的值在创建后是不可被变更的。每次在对字符串进行操作的时候(例如运用字符串的"+"操作),unity会新建一个字符串用来存储新的字符串,使得旧的字符串被废弃,这样就会造成内存垃圾。

我们可以采用以下的一些方法来最小化字符串的影响:

1.减少不必要的字符串的创建,如果一个字符串被多次利用,我们可以创建并缓存该字符串。

2.减少不必要的字符串操作,例如如果在Text组件中,有一部分字符串需要经常改变,但是其他部分不会,则我们可以将其分为两个Text组件,对于不变的部分就设置为类似常量字符串即可。

3.如果我们需要实时的创建字符串,我们可以采用StringBuilderClass来代替,StringBuilder专为不需要进行内存分配而设计,从而减少字符串产生的内存垃圾。

4.移除游戏中的Debug.Log()函数的代码,尽管该函数可能输出为空,对该函数的调用依然会执行,该函数会创建至少一个字符(空字符)的字符串。如果游戏中有大量的该函数的调用,这会造成内存垃圾的增加。

 

2.Unity函数调用

在代码编程中,当我们调用不是我们自己编写的代码,无论是Unity自带的还是插件中的,我们都可能会产生内存垃圾。Unity的某些函数调用会产生内存垃圾,我们在使用的时候需要注意它的使用。

这儿没有明确的列表指出哪些函数需要注意,每个函数在不同的情况下有不同的使用,所以最好仔细地分析游戏,定位内存垃圾的产生原因以及如何解决问题。有时候缓存是一种有效的办法,有时候尽量降低函数的调用频率是一种办法,有时候用其他函数来重构代码是一种办法。现在来分析unity中常见的造成堆内存分配的函数调用。

在Unity中如果函数需要返回一个数组,则一个新的数组会被分配出来用作结果返回,这不容易被注意到,特别是如果该函数含有迭代器,下面的代码中对于每个迭代器都会产生一个新的数组:

void ExampleFunction() {
    for(int i = 0; i < myMesh.normals.Length; i++) {
        Vector3 normal = myMesh.normals[i];
    }
}

对于这样的问题,我们可以缓存一个数组的引用,这样只需要分配一个数组就可以实现相同的功能,从而减少内存垃圾的产生:

void ExampleFunction() {
    Vector3[] meshNormals = myMesh.normals;
    for(int i = 0; i < meshNormals.Length; i++) {
        Vector3 normal = meshNormals[i];
    }
}

此外另外的一个函数调用GameObject.name 或者 GameObject.tag也会造成预想不到的堆内存分配,这两个函数都会将结果存为新的字符串返回,这就会造成不必要的内存垃圾,对结果进行缓存是一种有效的办法,但是在Unity中都对应的有相关的函数来替代。对于比较gameObject的tag,可以采用GameObject.CompareTag()来替代。除此之外我们还可以用Input.GetTouch()和Input.touchCount()来代替Input.touches,或者用Physics.SphereCastNonAlloc()来代替Physics.SphereCastAll()。

 

3.装箱拆箱操作

装箱操作是指一个值类型变量被用作引用类型变量时候的内部变换过程,如果我们向带有对象类型参数的函数传入值类型,这就会触发装箱操作。比如String.Format()函数需要传入字符串和对象类型参数,如果传入字符串和int类型数据,就会触发装箱操作。

 

4.协程

调用 StartCoroutine()会产生少量的内存垃圾,因为unity会生成实体来管理协程。所以在游戏的关键时刻应该限制该函数的调用。基于此,任何在游戏关键时刻调用的协程都需要特别的注意,特别是包含延迟回调的协程。

yield在协程中不会产生堆内存分配,但是如果yield带有参数返回,则会造成不必要的内存垃圾,例如:

yield return 0;

由于需要返回0,引发了装箱操作,所以会产生内存垃圾。这种情况下,为了避免内存垃圾,我们可以这样返回:

yield return null;

另外一种对协程的错误使用是每次返回的时候都new同一个变量,例如:

while(!isComplete) {
    yield return new WaitForSeconds(1f);
}

我们可以采用缓存来避免这样的内存垃圾产生:

WaitForSeconds delay = new WaiForSeconds(1f);
while(!isComplete) {
    yield return delay;
}

如果游戏中的协程产生了内存垃圾,我们可以考虑用其他的方式来替代协程。重构代码对于游戏而言十分复杂,但是对于协程而言我们也可以注意一些常见的操作,比如如果用协程来管理时间,最好在update函数中保持对时间的记录。如果用协程来控制游戏中事件的发生顺序,最好对于不同事件之间有一定的信息通信的方式。对于协程而言没有适合各种情况的方法,只有根据具体的代码来选择最好的解决办法。

 

5.函数引用

函数的引用,无论是指向匿名函数还是显式函数,在unity中都是引用类型变量,这都会在堆内存上进行分配。匿名函数的调用完成后都会增加内存的使用和堆内存的分配。具体函数的引用和终止都取决于操作平台和编译器设置,但是如果想减少GC最好减少函数的引用。

 

6.LINQ和常量表达式

由于LINQ和常量表达式以装箱的方式实现,所以在使用的时候最好进行性能测试。

 

重构代码来减小GC的影响

即使我们减小了代码在堆内存上的分配操作,代码也会增加GC的工作量。最常见的增加GC工作量的方式是让其检查它不必检查的对象。struct是值类型的变量,但是如果struct中包含有引用类型的变量,那么GC就必须检测整个struct。如果这样的操作很多,那么GC的工作量就大大增加。在下面的例子中struct包含一个string,那么整个struct都必须在GC中被检查:

public struct ItemData {
    public string name;
    public int cost;
    public Vector3 position;
}
ItemData[] itemData;

我们可以将该struct拆分为多个数组的形式,从而减小GC的工作量:

string[] itemNames;
int[] itemCosts;
Vector3[] itemPositions;

 

另外一种在代码中增加GC工作量的方式是保存不必要的Object引用,在进行GC操作的时候会对堆内存上的object引用进行检查,越少的引用就意味着越少的检查工作量。在下面的例子中,当前的对话框中包含一个对下一个对话框引用,这就使得GC的时候会去检查下一个对象框:

public class DialogData {
     DialogData nextDialog;
     public DialogData GetNextDialog() {
           return nextDialog;
     }
}

通过重构代码,我们可以返回下一个对话框实体的标记,而不是对话框实体本身,这样就没有多余的object引用,从而减少GC的工作量:

public class DialogData {
    int nextDialogID;
    public int GetNextDialogID() {
       return nextDialogID;
    }
}

 

定时执行GC操作

如果我们知道堆内存在被分配后并没有被使用,我们希望可以主动地调用GC操作,或者在GC操作并不影响游戏体验的时候(例如场景切换的时候),我们可以主动的调用GC操作:

System.GC.Collect()

 

补充

下面是一些个人知道的可以提高性能的一些注意点。有什么不对或者漏的欢迎大家补充

1.减少.Count .Length的调用

for(int i = 0; i < list.Count; i++) {
    //do something
}

应改为:

for(int i = 0, j = list.Count; i < j ; i++) {
    //do something
}

 

2.减少gameobject,transform,GetComponent的使用

我们可以在Start()方法中预先存储好这些值,之后使用的时候调用预先存储好的值即可。

GameObject m_gameObject;
Transform m_transform;
Text m_text;

void Start () {
    m_gameObject = gameObject;
    m_transform = transform;
    m_text = GetComponent();
}

 

3.减少SetActive(bool)的使用

对于要频繁显示隐藏的物体,我们可以减少使用SetActive(bool),而是通过transform.position信息将其移除视线外,例如UI的隐藏。

 

你可能感兴趣的:(Unity)