GC.Collect()

GC.Collect()其功能就是强制对所有代进行垃圾回收

垃圾的产生

class Person{
	String name;     //人员的姓名
	int age;       //人的年龄
	public void tell(){
		System.out.println("姓名:" + name + "、年龄:" + age);
	}
}
public class JavaDemo{
	public static void main(String args[]){
		Person per1 = new Person();   //声明并实例化对象
		Person per2 = new Person();
		
		per1.name = "张三";
		per1.age = 18;
		
		per2.name = "李四";
		per2.age = 19;
		
		per2 = per1;
		per2.age = 80;
		per1.tell();  //方法的调用
	}
}

此时完成了引用传递,并且也成功的完成了引用传递的处理操作,但是下面来观察一下其内存的分配与处理流程
一个栈内存只能够保存有一个堆内存的地址数据,如果发生更改,则之前的地址数据将从此栈内存中彻底消失。
在这里插入图片描述
所谓的垃圾空间指的就是没有任何栈内存所指向的对内存空间,所有的垃圾将被GC(Garbage Collector、垃圾收集器)不定期进行回收并且 释放无用内存空间,但是如果垃圾过多,一定将影响到GC的处理性能,从而降低整体的程序性能,那么在实际开发之中,对于垃圾的产生越少越好。

游戏运行时使用内存来存储数据,当这些数据不再被使用时,存储这些数据的内存被释放以便于之后这些内存可以被复用。垃圾(Garbage )是存储无用数据的内存的术语,GC(Garbage Collection 垃圾回收)是使这些内存可以再次使用的过程。

GC是Unity管理内存的一部分,我们的游戏可能因为GC负担过重而表现不佳,所以GC是引起性能问题的一个常见原因。

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

  • Unity可以访问两个内存池:(也称为托管堆)。栈用于短期存储小块数据,堆用于长期存储和较大数据段。
  • 当创建变量时,Unity从栈或堆中申请内存
  • 只要变量在作用域内(仍然可以通过我们的代码访问),分配给它的内存仍然在使用中, 我们称这部分内存已被分配。 我们将栈中的变量称为栈对象,将堆中的变量称为堆对象
  • 当变量超出作用域,该内存不再被使用并可以归还给原来的内存池。当内存被归还给原有的内存池里,我们称该内存被释放。栈内存在变量超出作用域时被实时释放,而堆内存在变量超出作用域之后并没有被释放并保持被分配的状态
  • 垃圾收集器(garbage collector)识别和释放未使用的堆内存。 垃圾收集器定期运行以清理堆。

现在我们了解事件的流程,让我们进一步了解栈分配和释放与堆分配和释放之间的区别。

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

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

  • 垃圾收集器检索堆上的每个对象。

  • 垃圾收集器搜索所有当前对象引用以确定堆上的对象是否仍在作用域内。

  • 不在作用域内的对象被标记为删除。

  • 删除被标记的对象并将内存返回给堆。

三种情况下会触发GC:

  • 堆分配时堆上的可用内存不足时触发GC。

  • GC会不时的自动运行(频率因平台而异)。

  • 手动强制调用GC

减少垃圾的产生量

可以使用一些技术来帮助我们减少代码中生成的垃圾量

缓存

如果我们的代码重复调用产生堆分配的函数,然后丢弃结果,这将产生不必要的垃圾。 对此,我们应该存储对这些对象的引用并复用它们。 这种技术被称为缓存

下面的函数每次调用都会引起堆分配,因为每次调用都会生成一个新的数组。

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

下面的代码只会有一次堆分配,因为数组创建赋值后被缓存起来了。缓存的数组可以复用因而不会产生垃圾。

private Renderer[] allRenderers;  

void Start()  {  allRenderers = FindObjectsOfType<Renderer>();  }  

void OnTriggerEnter(Collider other)  {  ExampleFunction(allRenderers);}

不要在频繁调用的函数中分配

如果我们需要在MonoBehaviour中分配堆内存,在频繁调用的函数里分配是最糟糕的。比如 每帧调用的函数 Update()和 LateUpdate(),在这些地方分配,垃圾将非常快的累积。我们应该尽可能在 Start() 或 Awake() 里缓存这些对象的引用,或者确保分配内存的代码只在需要的时候被运行。

让我们来看个简单的例子,下面的代码在 每次 Update()调用时都会调用一个引起堆分配的函数,会非常快的产生垃圾

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

简单修改后,可以确保产生堆分配的函数只在 transform.position.x 的值改变时才被调用.这样只在需要的时候产生堆分配而不会每帧都产生.

private float previousTransformPositionX;  
void Update()  
{  

  float transformPositionX = transform.position.x;  
  if (transformPositionX != previousTransformPositionX)  
  {  
    ExampleGarbageGeneratingFunction(transformPositionX);  
    previousTransformPositionX = transformPositionX;  
  }
}

另一个在 Update() 函数中减少垃圾内存产生量的方法是使用计时器.这适用于那些会产生垃圾内存的代码需要被频繁调用又不需要每帧调用的地方

下面的示例代码,产生垃圾内存的函数每帧被调用

void Update(){  ExampleGarbageGeneratingFunction();}

下面的代码,使用一个计时器来保证产生垃圾内存的函数每秒只被调一次

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

像这样对频繁调用函数的小改动,可以显著的减少垃圾内存的产生量

清空容器

创建容器类会引起堆分配,如果在代码中发现多次创建同一个容器变量,则应该缓存该容器引用并在重复创建的地方使用 Clear() 操作来替代

下面的示例中每次 *new *操作都会产生一次堆分配

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

下面的示例中,只在容器被创建 或者扩容时才 会有堆分配,显著减少了垃圾内存的产生量

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

对象池

即使减少了脚本中的堆分配,在运行时大量对象的创建和销毁依然会引起GC问题. 对象池是一种通过重用对象而不是重复创建和销毁对象来减少分配和释放的技术.对象池在游戏中广泛使用,最适合于频繁产生和销毁类似对象的情况;,例如,当枪射击子弹时.

对象池的完整指南超出了本文的范围,但它是一个非常有用的技术,值得一试. 关于Unity学习网站上的对象池的 这个教程 是在Unity中实现对象池系统的一个很好的指导

引起不必要堆分配的常见原因

我们知道局部的,值类型的变量被分配在栈上,其他的都在堆上分配.但是很多情况下的堆分配可能让人惊讶.我们来看看一些不必要的堆分配的常见原因,并考虑如何最好地减少这些。

字符串

在C#中, 字符串 是引用类型,而不是值类型,尽管它们似乎保持字符串的“值”. 这意味着创建和丢弃字符串会产生垃圾.由于字符串常用在很多代码中,所以这些垃圾可能累积。

C#中的字符串也是不可变的,这意味着它们的值在第一次创建之后不能再被更改。 每次我们操纵一个字符串(例如,通过使用+运算符来连接两个字符串),Unity将创建一个包含更新值的新字符串,并丢弃旧字符串。 这会产生垃圾。

我们可以遵循一些简单的规则,将字符串产生的垃圾减至最少。 我们来看看这些规则,然后看一下应用它们的例子。

  • 减少不必要的字符串创建。 如果多次使用相同的字符串值,应该创建一次该字符串并缓存该值。

  • 减少不必要的字符串操作。 例如,如果有一个经常更新的Text组件,并且包含一个连接的字符串,可以考虑将它分成两个Text组件。

  • 如果必须在运行时构建字符串,应该使用StringBuilder类。 StringBuilder类用于创建没有堆分配的字符串,并且在连接复杂字符串时减少生成的垃圾量。

  • 当不在需要调试时,立即删除对Debug.Log()的调用。即使没有输出任何内容,对Debug.Log()的调用依然会被执行。调用Debug.Log() 创建和处理至少一个字符串,所以如果我们的游戏包含许多这些调用,垃圾会累积

来看一个低效使用字符串而产生不必要垃圾的代码的例子。 在下面的代码中,在 Update()中创建一个连接 “TIME:”与浮点计时器的值的字符串来显示分数,这产生了不必要的垃圾。

public Text timerText;  
private float timer;  
void Update()  
{  
  timer += Time.deltaTime;  
  timerText.text = "TIME:" + timer.ToString();
}

下面我们做些改进。 我们把单词“TIME:”放在一个单独的文本组件中,并在Start()中设置它的值。 这样在Update()中,我们不再需要连接字符串。 可以大大减少垃圾的产生。

public Text timerHeaderText;  
public Text timerValueText;  
private float timer;  
void Start()  {    timerHeaderText.text = "TIME:";  }  
void Update()  {  timerValueText.text = timer.toString();}

Unity函数调用

重要的是要注意,每当我们调用不是自己写的代码时,无论是在Unity中还是在插件中,都可能会产生垃圾。 调用 一些Unity函数会产生堆分配,因此应谨慎使用以避免产生不必要的垃圾。

并没有一个应该避免使用的函数列表。 每个函数在某些情况下都是有用的,而在其他情况下则不太有用。所以最好仔细分析我们的游戏,确定垃圾的产生位置并仔细思考如何处理。 在某些情况下,可以缓存函数的结果; 在某些情况下,可以降低调用函数的频率; 在其他情况下,最好重构代码以使用不同的函数。 话虽如此,我们来看几个常见的 会导致堆分配 的 Unity函数,并考虑如何更好地处理它们。

每次访问返回值为数组的Unity函数时,都会创建一个新的数组,并将其作为返回值传递给我们。 这种行为并不总是显而易见的或可预期的,特别是当函数是 访问器 的时候(例如 Mesh.normals )。

下面的代码中,每次循环迭代都会生成一个新的数组

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

这种情况下很容易减少分配:我们可以简单地缓存对数组的引用。 这样可以只创建一个数组,并相应地减少了产生的垃圾量。

下面的代码演示了这一点。 在这种情况下,我们在循环之前调用Mesh.normals并缓存引用,这样就只创建一个数组。

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

访问GameObject.name或GameObject.tag也会有堆分配。 这两个都是返回新字符串的访问器,这意味着调用这些函数会产生垃圾。 缓存该值可能是有用的,但在这种情况下,可以使用相关的Unity函数。 要检查一个GameObject的标签的值而不产生垃圾,我们可以使用 GameObject.CompareTag()

下面的示例代码中,访问 GameObject.tag 会产生垃圾内存:

private string playerTag = "Player";  
void OnTriggerEnter(Collider other)  {  bool isPlayer = other.gameObject.tag == playerTag;}

如果使用 GameObject.CompareTag() ,则该函数不会产生垃圾:

private string playerTag = "Player";  
void OnTriggerEnter(Collider other)  {  bool isPlayer = other.gameObject.CompareTag(playerTag);}

GameObject.CompareTag并不是唯一的,很多Unity的函数都有无堆分配的替代版本。比如可以使用 Input.GetTouch()Input.touchCount 替换 Input.touches, 或者使用 Physics.SphereCastNonAlloc() 替换 Physics.SphereCastAll()

装箱

装箱 是指当一个值类型变量被用作一个引用类型变量时所执行的操作。当我们将值类型的变量(如int或float)传递给具有object类型参数的函数时,通常会发生装箱,如Object.Equals()函数。

例如,函数String.Format()接受一个string和一个object参数。 当我们传递一个string和一个int时,int就会被装箱。 下面的代码包含了一个装箱的例子:

void ExampleFunction()  
{  
    int cost = 5;  
    string displayString = String.Format("Price: {0} gold", cost);
}

装箱会产生垃圾源于其后台操作。当一个值类型变量被装箱时,Unity在堆上创建一个临时的System.Object来包装值类型变量。 一个System.Object是一个引用类型的变量,所以当这个临时对象被处理掉时会产生垃圾。

装箱是不必要的堆分配的常见原因。 即使我们不在我们的代码中直接装箱变量,我们可能也会使用导致装箱的插件,装箱也可能发生在其他函数的后台。 最好的做法是尽可能避免装箱,并删除导致装箱的任何函数调用。

协程

调用StartCoroutine()会产生少量的垃圾,因为Unity必须创建一些管理协程的实例的类。 所以,当游戏在交互时或在性能热点时应该限制对StartCoroutine()的调用。 为了减少这种方式产生的垃圾,必须在性能热点运行的协程应该提前启动,当使用可能包含对StartCoroutine()的延迟调用的嵌套协程时,我们应特别小心。

协程中的yield语句不会自己产生堆分配; 然而,我们传递给yield语句的值可能会产生不必要的堆分配。 例如,以下代码会产生垃圾:

yield return 0;

该代码产生垃圾,因为int变量0被装箱。 在这种情况下,如果我们希望只是等待一个帧而不会导致任何堆分配,那么最好的方法是使用以下代码:

yield return null;

协程的另一个常见错误是在多次使用相同的值时使用了new操作, 例如,以下代码将在循环迭代时每次都重复创建和销毁一个WaitForSeconds对象:

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

如果缓存和复用 WaitForSeconds 对象,就能减少垃圾的产生量,请看以下示例代码:

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

如果我们的代码由于协程而产生大量垃圾,我们可能考虑 使用除协程之外的其他东西来 重构我们的代码。 重构代码是一个复杂的问题,每个项目都是独一无二的,但是有一些常用的手段或许对协程问题有帮助。 例如,如果我们主要使用协同程序来管理时间,我们可以简单地在一个Update()函数中记录时间。 如果我们主要使用协同程序来控制游戏中发生的事情的顺序,我们可以创建某种消息系统来允许对象进行通信。 一个方法不能解决所有问题,但是有必要记住,在代码中可以有多种方法来实现相同的事情。

foreach循环

在Unity5.5之前的版本中,使用foreach遍历数组之外的所有集合,在循环终止时都会产生垃圾,这是因为其后台的装箱操作。当循环开始并且循环终止时,一个System.Object对象被分配在堆上。 Unity 5.5中已修复此问题。

在5.5之前的Unity版本中,以下代码中的循环会生成垃圾:

void ExampleFunction(List listOfInts)  
{  
    foreach (int currentInt in listOfInts)  
    {  
        DoSomething(currentInt);  
    }
}

如果我们无法升级我们的Unity版本,则有一个简单的解决方案来解决这个问题。 for和while循环不会在后台引起装箱,因此不会产生任何垃圾。 当迭代不是数组的集合时,我们应该优先使用它们。

下面的代码不会产生垃圾:

void ExampleFunction(List listOfInts)  
{  
    for (int i = 0; i < listOfInts.Count; i ++)  
    {  
        int currentInt = listOfInts[i];  
        DoSomething(currentInt);  
    }

}

函数引用

函数引用,无论是引用 匿名函数 还是命名函数,都是Unity中的引用类型变量。 它们将导致堆分配。 将匿名函数转换为 闭包( 匿名函数可在其创建时访问范围中的变量)显著增加了内存使用量和堆分配数量。

函数引用和闭包如何分配内存的精确细节因平台和编译器设置而异,但是如果GC是一个问题,那么最好在游戏过程中尽量减少使用函数引用和闭包。 这个Unity性能最佳实践指南 在这个主题上有更多的技术细节。

LINQ和正则表达式

LINQ和正则表达式由于在后台会有装箱操作而产生垃圾。在有性能要求的时候最好不使用。 同样, 这个Unity性能最佳实践指南 提供了有关此主题的更多技术细节。

构建代码以最小化GC的影响

代码的构建方式可能会影响GC。即使代码中没有堆分配,也有可能增加GC的负担。

可能增加GC的负担之一是要求它检查它不应该检查的东西。Structs是值类型变量,但是如果有一个包含引用类型变量的struct,那么垃圾收集器必须检查整个结构体。 如果有大量这样的结构体,那么垃圾回收器将增加大量额外的工作。

在这个例子中,下面的struct包含了一个引用类型的字符串。 现在 在垃圾回收器运行时 必须 检查 结构体的整个数组 。

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

private ItemData[] itemData;

在这个例子中,我们将数据存储在单独的数组中。 当垃圾收集器运行时,它只需要检查字符串数组,并且可以忽略其他数组。 这减少了垃圾收集器的工作。

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

另一个可能增加GC负担的操作是使用不必要的对象引用,当垃圾收集器搜索对堆上对象的引用时,它必须检查代码中的每个当前对象引用。 更少的对象引用意味着更少的工作量,即使我们不减少堆上的对象总数。

在这个例子中,我们有一个类填充一个对话框。 当用户查看对话框时,会显示另一个对话框。 我们的代码包含对应该显示的DialogData的下一个实例的引用,这意味着垃圾回收器必须在其操作中检查此引用:

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

这里我们重构下代码,以便它返回一个用于查找下一个DialogData实例的标识符,而不是实例本身。 这不是一个对象引用,所以它不会增加垃圾收集器所花费的时间。

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

这是个小例子。 然而,如果我们的游戏中有许多包含 对其他对象引用的 对象,那么我们可以通过以这种方式重构代码来大大降低堆的复杂性。

定时GC

手动强制GC

最后,我们可能希望自己触发GC。 如果我们知道堆内存已被分配但不再使用(例如,如果我们的代码在加载资源时生成垃圾),并且我们知道垃圾收集冻结不会影响播放器(例如,当加载界面还显示时),我们可以使用以下代码请求GC:

  • 7
    点赞
  • 32
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

爱游戏开发的蝎子

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值