Unity的GC优化原理及实践_unity 增量 gc-程序员宅基地

技术标签: unity  游戏引擎  

1.概述

1.1 简介

内存管理一直都是一个让人比较头疼的东西,尤其是现在重度游戏越来越多,每一次卡顿、每一次内存增长对玩家来说都是一个比较差的体验。技术群里总是有人调侃,游戏开发久了人就会变成“GC怪”。事实上,在游戏开发过程中,随着功能的不停迭代,内存问题一直都不能松懈。 Unity 2018集成了正式版的 .NET 4.x 和 C#7.3 ,引入了ref return和ref locals,让值类型操作更加高效,UnsafeUtility让Unsafe编程和Native Memory操作更加方便。Unity 2019加入了增量式GC,减少了GC带来的卡顿问题。目前来说,虽然内存管理还是一个需要注意的问题,但是却比以往版本灵活易用了很多。 本文希望可以从原理出发,以逐步递进的方式讲解GC优化的问题,主要关注于逻辑代码方面,希望可以给大家带来一定的参考价值。

1.2 什么是GC

GC的全称是Garbage Collection,也就是垃圾回收,是一种自动管理堆内存的机制,管理堆内存上对象的分配和释放。 一般来说,程序比较常用的有三种内存管理方式。 第一种是手动管理,即像C/C++一样使用malloc/free或者new/delete来为对象分配释放内存。这种管理方法的优点就是速度快,没有任何额外的开销,缺点是必须追踪每一个对象的使用情况,很容易发生各种问题,比如内存泄漏、野指针和空悬指针等。 第二种方法是使用引用计数(Reference Count)。它的思想是对象创建出来后,维护一个针对该对象的计数,使用该对象的地方对该计数加1,使用完毕后再减1,当计数为0时,销毁该对象。这种方法可以看做是一种半自动的内存管理方式,优点是可以把分配和释放的开销分布在实际使用过程当中,速度也比较快,不过会存在一个循环引用的问题。引用计数是一种比较常用的内存管理方法,比如Unity中的物理引擎PhysX就是使用引用计数来管理各种对象的。 最后一种方法是本文的重点,即追踪式GC器(Tracing Garbage Collector),Unity使用的GC器是一种叫标记/清除(Mark/Sweep)的算法,它的思路是当程序需要进行垃圾回收时,从根(GC Root)出发标记所有可达对象,然后回收没有标记的对象,这是一种全自动的内存管理方法,程序员完全不用追踪对象的使用情况,也不存在循环引用无法回收的问题,而在Unity中,使用的是一种叫Boehm-Demers-Weiser的GC器,它有以下特点: Stop The World:即当发生GC时,程序的所有线程都必须停止工作,等GC完成才能继续,Unity不支持多线程GC,即使是Unity 2019后使用的增量式GC,在回收时也是要停掉所有线程。 不分代:.NET和Java会把托管堆分成多个代(Generation),新生代的内存空间非常小,而且一般来说,GC主要会集中在新生代上,这让每一次GC的速度也非常快,但是Unity的GC是完全不分代的,即只要发生GC,就会对整个托管堆进行GC(Full GC)。 不压缩:不会对堆内存进行碎片整理,如下图: 图片来源于Unity的官方示例图

不分代:

不压缩:

GC会造成托管堆出现很多这样的空白“间隙”,这些间隙不会合并,当申请一个新对象时,如果没有任何一个间隙大于这个新对象大小,堆内存就会增加。

1.2 为什么要优化GC

优化GC是指降低内存开销,而不是指垃圾回收的过程。

1.GC的触发会导致进程锁死,GC的内存越多,锁死的时间越多.

2.进程自动触发GC,具有不可控性.

1.3 影响GC性能的主要因素

影响GC速度的因素主要有两个: 可达对象数量 托管堆的大小 可达对象是指不会当次GC被回收的对象,减少此类开销的主要方法就是减少对象数量,参考以下实现方法:

class Item
{
      public int a;
      public short b;
}
Item[] items;

对于items数组,每一个元素都会产生一个对象。 而以下代码,不管a和b有多少个元素,数组都只有一个对象,这样就会减少对象数量。

class Item
{
      public int[] a;
      public short[] b;
}
Item item;

而优化托管堆大小主要通过以下几个方面: 减少临时分配:临时分配的内存会产生碎片。 减少内存泄漏:即再也用不到但是又因为存在对其引用无法回收的对象。

1.4 什么时候会触发GC

1.第一代(第0代?)内存不够的时候

2.系统内存不足

3.主动调用GC.Collect()(不是必定会触发)

1.5 容易产生GC的地方以及解决办法

1. New obj

通过复用内存解决:类型池,对象池,容器池。

 2. Boxing(装箱)

如:Dictionary使用枚举或者自定义结构体当做Key

 3. String操作

字符串操作带来的GC很难避免,应尽量避免使用gameObject.name和gameObject.tag。常用字符串使用全局字符串常量。高频率拼接显示字符串的界面可以考虑用ZString,比如大量倒计时界面。

4.匿名函数和闭包

第一次匿名函数会有124B的GC,引用了外部变量每次都会GC,但是并不好避免经常会在回调里面写匿名函数。

 5.委托滥用

同一个委托+=操作触发GC,需要将原来的内存拼接上新的内存,整块移动到新的内存块,GC成指数型增长。

 执行第一次:

 执行第二次:

6. 一些语法糖

2.类和结构

类和结构的区别以及装箱和拆箱,基本上都是老生常谈了,不过,在开发过程中,还是会产生一个疑问:我的数据该使用类还是结构?这个问题接下来的几个部分都会有涉及到。

2.1 如何估算对象和结构体的大小

结构是值类型,它的结构体实例是存放在栈或者堆中的。在栈中我们保有的是实例的值,所以每一次赋值,都会在栈中多赋值一份实例出来。结构体在内存中所占大小,就是其字段所占大小,但是,结构体的大小并不是简单的所有字段的大小相加,而是存在一个对齐规则,在默认的对齐规则中,基本类型字段是按照自身大小对齐的,如byte是按1字节对齐,int是按4字节对齐。如下面的结构体:

  struct S
  {
      byte b1;
  }

这个结构体的大小是1,如果在下面添加一个字段:

  struct S
  {
      byte b1;
      int i1;
  }

这个结构体的大小是8,因为int是4字节对齐的,所以只能从第四个字节开始。 如果再添加一个字段:

  struct S
  {
      byte b1;
      int i1;
      byte b2;
  }

这个结构体的大小是12,由于struct本身也是要对齐的,所以它的对齐规则是按照其中元素最大的对齐规则决定的。如当前这个结构体是按照i1的对齐规则决定的,也就是四字节对齐,不足四字节则不齐。如果想优化其大小,调整顺序如下,结构体的大小就变成了8。

  struct S
  {
      byte b1;
      byte b2;
      int i1;
  }

类是引用类型,它的对象实例存放在堆中,对象实例一定是会占用堆内存的,而在栈中,我们保有的是实例的引用,对象在堆内存中大概是如下图所示: 其中vtable是类的共有数据,包含静态变量和方法表(在Mono中,结构的静态变量也存放在vtable里,它是缓存在一个叫tablecache的哈希表当中的,而IL2CPP中类和结构的静态变量存在一个单独的类里)。Monitor是线程同步用的,这两个指针分别占用一个IntPtr.Size大小(32位中是4字节,64位中是8字节),再下面是所有字段,字段是从第9个字节或17个字节开始的,字段的对齐规则与结构体的对齐规则相同,区别是Mono中对象实例会把引用类型的引用摆在最前面。一个对象实例的大小(instance_size)就是IntPtr.Size * 2+字段所占大小,结构体被装箱后在堆内存的大小也一样。 通过调整字段顺序,可以优化对象和结构体大小,特别是有容器存放多个对象或结构体的,可以减少堆内存占用。 此外,我们还可以通过StructLayoutAttribute自定义类和结构字段的对齐方式。比如下面的结构体:

  [StructLayout(LayoutKind.Sequential, Pack = 1)]
  public struct S
  {
      byte b1;
      int i1;
      byte b2;
  }

该结构体强制按1字节对齐,所以它的大小就是6。

  [StructLayout(LayoutKind.Explicit)]
  public struct S
  {
      [FieldOffset(0)]byte b1;
      [FieldOffset(0)]int i1;
      [FieldOffset(1)] byte b2;
  }

这个结构体的大小是4,它实现了类似C/C++中union的类型,b1、b2与i1共用同一段内存,b1和b2也代表了i1的前两个字节。 注意,内存对齐是会考虑硬件优化的,使用StructLayout修改对齐方式有可能会降低性能。

2.2 装箱和拆箱

装箱和拆箱的过程很多文档都会有描述,这里就不再细说了。只说几个比较

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/jmu201521121014/article/details/107858015

智能推荐

RK3566开发板编译安卓源码问题及解决_openharmony rk3566 编译-程序员宅基地

文章浏览阅读2.9k次,点赞2次,收藏3次。第一次编译安卓,记录一下Ubuntu版本问题:这个问题困扰了我三天。。。才疏学浅。。。使用Ubuntu20及以上版本可能会出现的问题及相关解决方法,虽然最后也没搞定。。。。Ubuntu20版本使用的是Python3,默认不支持Python2,所以在编译时使用Python2就要单独安装1.sudo apt-get install python-2 会安装Python2,但不是默认使用的,需要定义为默认使用(方法百度。。),且安装pip总出错。。。可能是我的问题。。所以转手动安装2.下载Pytho_openharmony rk3566 编译

UE4 读取本地图片_ue 通过路径获取图片-程序员宅基地

文章浏览阅读6.9k次,点赞4次,收藏6次。参考链接:https://answers.unrealengine.com/questions/235086/texture-2d-shows-wrong-colors-from-jpeg-on-html5-p.html我这里,不能将图片全放工程之中,需要在外部在加载图片资源,再来使用1.通过本地图片路径,获取图片,并将其数据转为uint类型的数组_ue 通过路径获取图片

通过OpenCV实现虚拟键盘_基于opencv的虚拟键盘-程序员宅基地

文章浏览阅读1.7k次,点赞6次,收藏22次。通过OpenCV实现虚拟键盘环境配置在Anaconda中创建一个python=3.7的环境可在pycharm内直接新建环境,也可以通过命令行的方式conda create -n your_env_name python=x.x激活或者切换虚拟环境 activate your_env_name然后对此虚拟环境中安装额外的包pip install cvzoneCollecting cvzone Using cached cvzone-1.4.1.tar.gz (11 kB)Collec_基于opencv的虚拟键盘

工具 | SEGGER 的RTT实时输出工具的使用_segger_rtt-程序员宅基地

文章浏览阅读3.3k次,点赞6次,收藏22次。一、前言在我们的嵌入式开发中,常常把printf重定向到MCU的串口外设,再配合上位机界面软件,通过打印调试信息的方式来调试我们的嵌入式软件。此处,我们介绍另一种打印调试的方法——SEGGER 的RTT 。RTT全称是Real Time Transmit(实时传输),是Segger公司推出的,是配合J-link使用的一种调试手段。其框图如下:可见,我们的MCU通过J-Link,凭借RTT就..._segger_rtt

微信小程序中justify-content: flex-end 失效_flex-end不生效-程序员宅基地

文章浏览阅读5.9k次,点赞2次,收藏2次。不正确的效果是这样的:页面wxml代码如下:<scroll-view class="hot-box" scroll-y="true"> <view class="orderDetails" wx:for="{{order}}" wx:key=" "> ........省略部分代码...... ........省略部..._flex-end不生效

Android音视频开发之音频录制和播放_androidshow-master录音-程序员宅基地

文章浏览阅读1.1k次。【代码】Android音视频开发之音频录制和播放。_androidshow-master录音

随便推点

C++ STL笔记_stl 笔记-程序员宅基地

文章浏览阅读385次,点赞12次,收藏7次。与类class的构造函数一样,结构体的构造函数必须是与结构体名称相同的公共成员函数,并且没有返回类型。而使用 struct 时,结构体中的成员默认都是 public 属性的。class 继承默认是 private 继承,而 struct 继承默认是 public 继承(《C++继承与派生》一章会讲解继承)。这就定义了一个Student类型的变量stu1,并且以列表的形式为其中的变量提供了初始值。C++中还可以使用构造函数来初始化结构体成员变量,这和初始化类class成员变量是相同的。_stl 笔记

什么是原型,proptotype和__proto__是什么?_.proptotype-程序员宅基地

文章浏览阅读357次,点赞8次,收藏9次。原型:原型是JavaScript中每个对象都有的一个特殊的属性。它指向的是另一个对象的引用,而这个对象就是原型对象。每个对象都是从原型对象中继承属性和方法的。在JavaScript中,所有的对象都是从Object构造函数的原型对象中继承而来的。prototype:是函数对象特有的一个属性,它指向该函数的原型对象。:是每个 JavaScript 对象都具有的一个属性,它指向该对象的原型对象。属性,我们可以访问到对象的原型链,即对象的原型链中的下一个对象。下面用一张图来解释一下原型、原型对象之间的关系。_.proptotype

照片dpi怎么调300在线处理?一键修改图片dpi_如何将低dpi图片升级到300dpi csdn-程序员宅基地

文章浏览阅读131次。在日常生活中,我们经常需要调整照片的分辨率,以满足不同需求,例如打印高质量的照片或在网上发布清晰的图像。而且生活用也有许多场景需要用到修改图片分辨率的地方,比如当您希望打印高质量的照片时,将照片的DPI设置为300可以确保打印输出具有清晰度和细节。通过使用在线工具调整照片的DPI为300,您可以提高照片的分辨率和图像质量,这种方法适用于多种生活场景,包括打印照片、网络发布和制作印刷品。点击上传图片,支持批量处理图片dpi,设置要修改DPI的数值,点击“开始生成”。完成后,点击下载图片。_如何将低dpi图片升级到300dpi csdn

MySQL实现把两行两列数据合并为一行一列_sql 两个一行一列-程序员宅基地

文章浏览阅读1.1w次,点赞2次,收藏5次。最近在oa项目中使用acitiviti中,遇到一个排他网关有多个判断条件(),并且可以多次执行,在显示已办任务的时候要归属为一条数据,利用GROUP_CONCAT和CONCAT加上group by 解决。详细sql如下:SELECTaht.ID_ AS id,ard.NAME_ AS processName,aht.NAME_ AS name ,art.NAME_ AS curre_sql 两个一行一列

在GitHub中的README.md文件中添加图片_readme.md怎么加图片-程序员宅基地

文章浏览阅读4.7k次。  有时候需要在GitHub中的README.md文件中添加图片或以优化阅读效果,github.com对README.md文件的预览提供了强大的支持,可以让我们很方便的实现这一功能。   首先需要把自己想要展示的图片上传到GitHub上,比如说:   创建之后,需要点击文件,会得到该图片的预览地址,注意,一定是这个预览地址,不能是代码仓的路径地址。   仔细观察这个地址,会有所不同..._readme.md怎么加图片

面试题:查询部门最高工资的员工信息_查询销售部工资最高的员工的姓名和工资-程序员宅基地

文章浏览阅读2.2w次。难度:中等表Employee保存了所有的员工数据。 Id Name Salary DepartmentId 1 Joe 70000 1 2 Henry 80000 2 3 Sam 60000 2 4 Max 90000 1  表Department保存了所有的部门数据。 Id Name 1 IT 2_查询销售部工资最高的员工的姓名和工资

推荐文章

热门文章

相关标签