Unity 之 UGUI 小总结

作者: 程序小妖精 | 来源:发表于2017-08-10 15:20 被阅读295次

   开发过程中对UGUI的一个小总结。

   首先从原画师拿到效果图,美术切图,拿到碎图后打成大图。

    我们先来说一下图:RGBA8888:每一个通道占8位。大图:1024*1024。高端    机:2048*2048。

   我们通常从美工那里拿来碎图,歪歪使用的一个工具texturepacker 把碎图打成大图,导出成 .tpsheet  .png格式。其次我们要做的是,在Unity3d中导入插件texture import(此插件会自动把大图打成图集).然后把大图导入U3D。

  在UI优化中较为明显而众所周知的就是降低DrawCall(注释:DrawCall CPU向GPU发送的一次渲染指令)。而降低DrawCall我们会采用静态合批和动态合批。(注释:合批必须同类型 Mesh 同材质。)DrawCall的标准 RGP 类 小于 150。

   然而,工作中我们优化大概分为UI层级计算,UI重建和多层级渲染。接下来我们来说一下:

UI层级 计算 :

1,计算层级:

1,如果有一个UI元素,它所占的屏幕范围内(通常是矩形),如果没有任何UI在它的底下,那么它的层级号就是0(最底下);

2,如果有一个UI在其底下且该UI可以和它Batch,那它的层级号与底下的UI层级一样;

3,如果有一个UI在其底下但是无法与它Batch,那它的层级号为底下的UI的层级+1;

4,如果有多个UI都在其下面,那么按前两种方式遍历计算所有的层级号,其中最大的那个作为自己的层级号。

合并批次原则;

同层级{同材质球,}

{0 : { image   , image, text  }}

1,Unity会将每一层的所有元素进行一个排序(按照材质、纹理等信息),合并掉可以Batch的元素成为一个批次

2,Text组件会排在Image组件之前渲染。

3,,Unity会再做一个优化,即如果相邻间的两个批次正好可以Batch的话就会进行Batch(这么处理,可以合成一个批次,如下:)

{0 : { text   image  }}textàimage

{1 : { text  ,image }}

0: textàimage    1: textàimage

下面我们来看一个列子,这样会更加的清晰:

一个层级为0的ImageA,一个层级为1的ImageB(2个Image可Batch)和一个层级为0的TextC,

textCàimageAàimageB

一个层级为0的TextD,一个层级为1的TextE(2个Text可Batch)和一个层级为0的ImageF,

0 : textD –>imageFà1 : textEàimage h    3:

0 : textD –>imageFà1 : textEàimage h

总结:

1,有相同材质和纹理的UI元素是可以Batch的,可以Batch的UI上下叠在一块不会影响性能,但是如果不能Batch的UI元素叠在一块,就会增加Drawcall开销

2,尽量让同一个材质球上的东西 放在同一级上

3,有些情况可以考虑人为增加层级从而减少Drawcall,比如一个Text的层级为0,另一个可Batch的Text叠在一个图片A上,层级为1,那此时2个Text因为层级不同会安排2个Drawcall,但如果在第一个Text下放一个透明的图片(与图片A可Batch),那两个Text的层级就一致了,Drawcall就可以减少一个。

Text0 –》Imageàtext 1

4,应该尽量避免使用Mask,其实Mask的功能有些时候可以变通实现,比如设计一个边框,让这个边框叠在最上面,底下的UI移动时,就会被这个边框遮住;

5, z值 保持为0

UI重建:

1,动静分离:

经常发生变动的ui单独放在一个Canvase

2,删除不必要的元素。

3,CanvasRender.setDisable .

4,Text的Best Fit选项.

5,Canvas的Pixel Perfect选项

6,使用缓存池来保存ScrollView中的Item,对于移出或移进View外的的元素,不要调用disable或enable,而是把它们放到缓存池里或从缓存池中取出复用。

7, 除了rebuild过程之外,UGUI的touch处理消耗也可能会成为性能热点。因为UGUI在默认情况下会对所有可见的Graphic组件调用raycast。对于不需要接收touch事件的grahic,一定要禁用raycast。对于unity5以上的可以关闭graphic的Raycast Target而对于unity4.6,可以给不需要接收touch的UI元素加上canvasgroup组件。

多层级渲染:

1,canvasRender .setdisable()

2,只挂载button替代  空的image

3,不要使用空的Image,在Unity中,RayCast使用Graphi作为基本元素来检测touch,在笔者参与的项目中,很多同学使用空的image并将alpha设置为0来接收touch事件,这样会产生不必要的overdraw。通过如下类NoDrawingRayCast来接收事件可以避免不必要的overdraw。

3. public class NoDrawingRayCast : Graphic

4. {

5.     public override void SetMaterialDirty()

6.     {

7.     }

8.     public override void SetVerticesDirty()

9.     {

10.     }

11.     protected override void OnFillVBO(List vbo)

12.     {

13.         vbo.Cslear();

14.     }

}

性能检测工具:

1profile

常见问题:

Q1:我在UGUI里更改了Image的Color属性,那么Canvas是否会重建?我只想借用它的Color做Animation里的变化量。

如果修改的是Image组件上的Color属性,其原理是修改顶点色,因此是会引起网格的Rebuild的(即Canvas.BuildBatch操作,同时也会有Canvas.SendWillRenderCanvases的开销)。而通过修改顶点色来实现UI元素变色的好处在于,修改顶点色可以保证其材质不变,因此不会产生额外的Draw Call。

Q2:Unity自带的UI Shader处理颜色时,改_Color属性不会触发顶点重建吗?

在UI的默认Shader中存在一个Tint Color的变量,正常情况下,该值为常数(1,1,1),且并不会被修改。如果是用脚本访问Image的Material,并修改其上的Tint Color属性时,对UI元素产生的网格信息并没有影响,因此就不会引起网格的Rebuild。但这样做因为修改了材质,所以会增加一个Draw Call。

Q:动静分离或者多Canvas带来性能提升的理论基础是什么呢?如果静态部分不变动,整个Canvas就不刷新了?

在UGUI中,网格的更新或重建(为了尽可能合并UI部分的DrawCall)是以Canvas为单位的,且只在其中的UI元素发生变动(位置、颜色等)时才会进行。因此,将动态UI元素与静态UI元素分离后,可以将动态UI元素的变化所引起的网格更新或重建所涉及到的范围变小,从而降低一定的开销。而静态UI元素所在的Canvas则不会出现网格更新和重建的开销。

Q:UWA建议“尽可能将静态UI元素和频繁变化的动态UI元素分开,存放于不同的Panel下。同时,对于不同频率的动态元素也建议存放于不同的Panel中。”那么请问,如果把特效放在Panel里面,需要把特效拆到动态的里面吗?

通常特效是指粒子系统,而粒子系统的渲染和UI是独立的,仅能通过Render Order来改变两者的渲染顺序,而粒子系统的变化并不会引起UI部分的重建,因此特效的放置并没有特殊的要求。

Q:多人同屏的时候,人物移动会使得头顶上的名字Mesh重组,从而导致较为严重的卡顿,请问一下是否有优化的办法?

如果是用UGUI开发的,当头顶文字数量较多时,确实很容易引起性能问题,可以考虑从以下几点入手进行优化:

1.尽可能避免使用UI/Effect,特别是Outline,会使得文本的Mesh增加4倍,导致UI重建开销明显增大;

2.拆分Canvas,将屏幕中所有的头顶文字进行分组,放在不同的Canvas下,一方面可以降低更新的频率(如果分组中没有文字移动,该组就不会重建),另一方面可以减小重建时涉及到的Mesh大小(重建是以Canvas为单位进行的);

3.降低移动中的文字的更新频率,可以考虑在文字移动的距离超过一个阈值时才真正进行位移,从而可以从概率上降低Canvas更新的频率。

三、界面切换

Q1:游戏中出现UI界面重叠,该怎么处理较好?比如当前有一个全屏显示的UI界面,点其中一个按钮会再起一个全屏界面,并把第一个UI界面盖住。我现在的做法是把被覆盖的界面SetActive(False),但发现后续SetActive(True)的时候会有GC.Alloc产生。这种情况下,希望既降低Batches又降低GC Alloc的话,有什么推荐的方案吗?

可以尝试通过添加一个Layer如OutUI, 且在Camera的Culling Mask中将其取消勾选(即不渲染该Layer)。从而在UI界面切换时,直接通过修改Canvas的Layer来实现“隐藏”。但需要注意事件的屏蔽,禁用动态的UI元素等等。

这种做法的优点在于切换时基本没有开销,也不会产生多余的Draw Call,但缺点在于“隐藏时”依然还会有一定的持续开销(通常不太大),而其对应的Mesh也会始终存在于内存中(通常也不太大)。

以上的方式可供参考,而性能影响依旧是需要视具体情况而定。

Q2:通过移动位置来隐藏UI界面,会使得被隐藏的UIPanel继续执行更新(LateUpdate有持续开销),那么如果打开的界面比较多,CPU的持续开销是否就会超过一次SetActive所带来的开销?

这确实是需要注意的,通过移动的方式“隐藏”的UI界面只适用于几个切换频率最高的界面,另外,如果“隐藏”的界面持续开销较高,可以考虑只把一部分Disable,这个可能就需要具体看界面的复杂度了。一般来说在没有UI元素变化的情况下,持续的Update开销是不太明显的。

Q3:如图,我们在UI打开或者移动到某处的时候经常会观测到CPU上的冲激,经过进一步观察发现是因为Instantiate产生了大量的GC。想请问下Instantiate是否应该产生GC呢?我们能否通过资源制作上的调整来避免这样的GC呢?如下图,因为一次性产生若干MB的GC在直观感受上还是很可观的。


准确的说这些GC Alloc并不是由Instantiate直接引起的,而是因为被实例化出来的组件会进行OnEnable操作,而在OnEnable操作中产生了GC,比如以上图中的函数为例:

上图中的Text.OnEnable是在实例化一个UI界面时,UI中的文本(即Text组件)进行了OnEnable操作,其中主要是初始化文本网格的信息(每个文字所在的网格顶点,UV,顶点色等等属性),而这些信息都是储存在数组中(即堆内存中),所以文本越多,堆内存开销越大。但这是不可避免的,只能尽量减少出现次数。

因此,我们不建议通过Instantiate/Destroy来处理切换频繁的UI界面,而是通过SetActive(true/false),甚至是直接移动UI的方式,以避免反复地造成堆内存开销。

四、加载相关

Q1:UGUI的图集操作中我们有这么一个问题,加载完一张图集后,使用这个方式获取其中一张图的信息:assetBundle.Load (subFile, typeof (Sprite)) as Sprite;这样会复制出一个新贴图(图集中的子图),不知道有什么办法可以不用复制新的子图,而是直接使用图集资源 。


经过测试,这确实是Unity在4.x版本中的一个缺陷,理论上这张“新贴图(图集中的子图)”是不需要的,并不应该加载。 因此,我们建议通过以下方法来绕过该问题:

在assetBundle.Load (subFile, typeof (Sprite)) as Sprite;之后,调用

Texture2D t = assetBundle.Load (subFile, typeof (Texture2D)) as Texture2D;

Resources.UnloadAsset(t);

从而卸载这部分多余的内存。

Q2:加载UI预制的时候,如果把特效放到预制里,会导致加载非常耗时。怎么优化这个加载时间呢?

UI和特效(粒子系统)的加载开销在多数项目中都占据较高的CPU耗时。UI界面的实例化和加载耗时主要由以下几个方面构成:

纹理资源加载耗时

UI界面加载的主要耗时开销,因为在其资源加载过程中,时常伴有大量较大分辨率的Atlas纹理加载,我们在之前的Unity加载模块深度分析之纹理篇有详细讲解。对此,我们建议研发团队在美术质量允许

1,1024*768    1024*1024

2, 640*480      512*512

相关文章

网友评论

    本文标题:Unity 之 UGUI 小总结

    本文链接:https://www.haomeiwen.com/subject/mmnylxtx.html