Unity学习笔记之——ugui的性能优化

发布于:2025-02-28 ⋅ 阅读:(12) ⋅ 点赞:(0)

在Unity中UI优化的核心问题就是重绘和批处理之间的平衡

一、Canvas优化要点

1.优化原因:

(1)Unity为了性能优化,会合并Canvas下的所有元素;

(2)如果把所有面板放到一个Canvas下,会造成重绘Redraw(反复绘制);

        下面列出了Unity中导致Canvas变脏的地方:

        · 设置顶点脏——SetVerticesDirty,如RectTransform、Image中各种参数修改等;

        · 设置材质脏——SetMaterialDirty,如Material、Texture相关修改等;

        · 设置布局脏——SetLayoutDirty,如Layout系列组件进行的修改等;

        · 设置上面三个都脏——SetAllDirty。

        所以说,对于UI,基本上只要动了就会触发Canvas重绘。

(3)在UGUI中,当Canvas中的元素发生任何变化(如主动切换、移动或调整大小等),都会运行一个过程(重建)来重建整个Canvas UI网格。重建过程的成本很高,如果执行太多次,或Canvas中的ui数量很大,性能就会受影响。

2.优化方法:

(1)一个Canvas下的所有UI元素都是合在一个Mesh中的,过大的Mesh在更新时开销很大,一般建议每个较为复杂的UI界面,都创建一个Canvas(可以是子Canvas),在UI界面很复杂时,甚至要划分更多的子Canvas,而Canvas又不能细分的太多,因为会导致DrawCall的上升;如果子画布中包含的元素发生变化(除了子画布中的UI被SetActive切换活动状态),则只会运行子画布的重建,而不会运行父画布。

(2)动静分离,理论基础是子Canvas不会导致父Canvas重绘。

        由于Canvas负责将几何图形合并成批,并且生成对应的渲染命令发送给Unity图形管线,所以动静分离虽然减少了Canvas重绘,但是增加了批次;

(3)把一个面板的UI资源放到一个图集里;

二、Mask遮罩

Mask可以在任何形状中被掏空,而RectMask2d只能被掏空为矩形。

用于剔除每帧的CPU负载与其屏蔽目标的增加成正比,所以要尽可能避免使用遮罩,即使使用了,在不需要时将enabled设置为false。

三、TextMeshPro

在TextMeshPro中设置文本的常用方法是将文本分配给text属性,但是还有另一个方法SetText。
例如,SetText有许多重载,它们接受字符串和float类型的值作为参数。

//1.这种方法减少了生成字符串的成本。
btnText.SetText("{0}", number);

//2.这种方法会执行float类型的ToString(),因此每次执行此过程都会产生字符串生成成本
btnText.text = number.ToString();

TextMeshPro的这个特性在与ZString 结合使用时也非常强大。
ZString是一个库,它减少了字符串生成过程中的内存分配。ZString为TMP_Text类型提供了许多扩展方法,通过使用这些方法,可以实现灵活的文本显示,同时减少字符串生成的成本。

四、UI显示开关

uGUI组件的特点是使用SetActive切换对象的高成本。这是由于OnEnable为各种重建设置Dirty标志并执行与掩码相关的初始化。因此,考虑使用SetActive方法的替代方法来切换UI的显示是很重要的。
第一种方法是将Canvas的enabled更改为false。这将阻止画布下的所有对象被渲染。因此,这种方法的缺点是,它只能在您想要隐藏Canvas下的所有对象时使用。

另一种方法是使用CanvasGroup。它有个函数可以调整它下面所有物体的透明度。如果你使用这个函数并将透明度设置为0,你可以隐藏其CanvasGroup

不过要注意这样处理虽然避免了由SetActive引起的负载,但gameobject会保持在活动状态,可能也会产生负载。

五、补充

非UI对象只要移出了视口,就不会进行批处理了。而UI对象需要将Canvas设置成WorldSpace模式或者Camera模式,而不是Overlay模式,在移出视口后才不会进行批处理。否则一旦使用Overlay模式,即使移出了视口也仍然会进行批处理。

一旦某个Canvas上存在一个UI在视口内(需要被渲染),则整个Canvas上的所有UI都需要被批处理。所以说要么把Canvas移出去很远(保证Canvas上的所有内容不渲染),要么不移出去(渲染Canvas上的所有内容)。