Unity 优化之UGUI (2017年版【一】)

首先感谢UWA的公开课讲解,本文中的内容主要采集于UWA2017的公开课公开课中的优化点对于项目的帮助是不容小觑的~

在去年观看uwa在直播公开课的时候,发现了多在自己使用ugui中使用不正确的地方,但是碍于项目工期比较紧张,并没有进行系统的整理,时间一久当时记下的东西就显得很碎片化,有些问题也变得模棱两可,趁着项目刚刚结尾,特意对公开课内容指出的注意事项,进行相应的总结、分类~

有不准确、不合理的地方希望大家能留言指出,方便及时更正

在公开课中的使用的Unity版本为4.6 目前本人实测版本为2017.4.2f2


尽可能的让UI元素合批

首先创建一个新的场景,摄像camera为Solod Color
Unity 优化之UGUI (2017年版【一】)_第1张图片
Unity 优化之UGUI (2017年版【一】)_第2张图片
可以看到场景中的Barches为1,这是因为camera在做Clear操作
Unity 优化之UGUI (2017年版【一】)_第3张图片
然后创建两个Button 去掉对应的Text,运行unity发现产生的Batches变为2
Unity 优化之UGUI (2017年版【一】)_第4张图片
当两个Button进行重叠的是时候,Batches并没有变化
Unity 优化之UGUI (2017年版【一】)_第5张图片
但是对有Text的两个Button进行上面的操作,就会出现不一样的效果,重叠后的Batches和未重叠Batches不一致
Unity 优化之UGUI (2017年版【一】)_第6张图片
Unity 优化之UGUI (2017年版【一】)_第7张图片

按照公开课中的说法,这种重叠的做法会打破原有的拓扑排序(Text2 与Button2合并),造成Draw Call的合并失败(Text -Button -Text - Button),当然在Hierarchy视图中也要尽量按照这种拓扑排序的方式进行UI的层级摆放。

结论:在进行UI设计额时候尽量少使用来自不同Atlas的材质,也尽量避免这种倾轧的情况(Scene视图调节为Writeframe模式可以更容易查看),尽可能把所有的文字放到图片之上,使用同种字体,更容易进行合批

Unity 优化之UGUI (2017年版【一】)_第8张图片

接下来要说的就是Mask组件

在一个新的场景中放图两个Image,然后设置统一的Packing Tag
Unity 优化之UGUI (2017年版【一】)_第9张图片
而且也要在Editor Settings开启图集选项
Unity 优化之UGUI (2017年版【一】)_第10张图片
运行前后对应可发现两张图片进行了合批
Unity 优化之UGUI (2017年版【一】)_第11张图片
Unity 优化之UGUI (2017年版【一】)_第12张图片
然后在场景中添加一个对应的Mask,发现Batches有所变化,两个图片也进行了合批
Unity 优化之UGUI (2017年版【一】)_第13张图片
把其中一个图片放到mask的遮罩中,发现图片无法合批
Unity 优化之UGUI (2017年版【一】)_第14张图片

结论: 首先一个Mask组件就会产生一个Draw Call,而且在Mask中的图片无法与外界的图片进行合批


减少Overdraw

Unity 优化之UGUI (2017年版【一】)_第15张图片
根据上图的显示,调节到OverDraw模式,颜色越鲜亮的地方造成的OverDraw越大,随之带来的GPU压力也是越大的

下面来说减少OverDraw的一些策略

在ImageType选项为Sliced的情况下,不需要Fill Center 的时候去掉勾选
Unity 优化之UGUI (2017年版【一】)_第16张图片
  • 作者钱康来重写Image相关组件降低性能消耗
Code Empty4Raycast
using UnityEngine;
using System.Collections;

namespace UnityEngine.UI
{
    public class Empty4Raycast : MaskableGraphic
    {
        protected Empty4Raycast()
        {
            useLegacyMeshGeneration = false;
        }

        protected override void OnPopulateMesh(VertexHelper toFill)
        {
            toFill.Clear();
        }
    }
}
Code PolygonImage
using System.Collections.Generic;

namespace UnityEngine.UI
{
    [AddComponentMenu("UI/Effects/PolygonImage", 16)]
    [RequireComponent(typeof(Image))]
    public class PolygonImage : BaseMeshEffect
    {
        protected PolygonImage()
        { }

        // GC Friendly
        private static Vector3[] fourCorners = new Vector3[4];
        private static UIVertex vertice = new UIVertex();
        private RectTransform rectTransform = null;
        private Image image = null;
        public override void ModifyMesh(VertexHelper vh)
        {
            if (!isActiveAndEnabled) return;

            if (rectTransform == null)
            {
                rectTransform = GetComponent();
            }
            if (image == null)
            {
                image = GetComponent();
            }
            if (image.type != Image.Type.Simple)
            {
                return;
            }
            Sprite sprite = image.overrideSprite;
            if (sprite == null || sprite.triangles.Length == 6)
            {
                // only 2 triangles
                return;
            }

            // Kanglai: at first I copy codes from Image.GetDrawingDimensions
            // to calculate Image's dimensions. But now for easy to read, I just take usage of corners.
            if (vh.currentVertCount != 4)
            {
                return;
            }

            rectTransform.GetLocalCorners(fourCorners);

            // Kanglai: recalculate vertices from Sprite!
            int len = sprite.vertices.Length;
            var vertices = new List(len);
            Vector2 Center = sprite.bounds.center;
            Vector2 invExtend = new Vector2(1 / sprite.bounds.size.x, 1 / sprite.bounds.size.y);
            for (int i = 0; i < len; i++)
            {
                // normalize
                float x = (sprite.vertices[i].x - Center.x) * invExtend.x + 0.5f;
                float y = (sprite.vertices[i].y - Center.y) * invExtend.y + 0.5f;
                // lerp to position
                vertice.position = new Vector2(Mathf.Lerp(fourCorners[0].x, fourCorners[2].x, x), Mathf.Lerp(fourCorners[0].y, fourCorners[2].y, y));
                vertice.color = image.color;
                vertice.uv0 = sprite.uv[i];
                vertices.Add(vertice);
            }

            len = sprite.triangles.Length;
            var triangles = new List(len);
            for (int i = 0; i < len; i++)
            {
                triangles.Add(sprite.triangles[i]);
            }

            vh.Clear();
            vh.AddUIVertexStream(vertices, triangles);
        }
    }
}

减少Raycast Target

在对应的Text Image和Rawimage中都有Raycast Target选项,这个选项负责接收我们所点击的事件,但是项目中我们有些UI元素是不需要这些东西,所有可以去掉,减少不必要的性能消耗,可以用 RaycastTarget检测小工具 ,他可以在对应的editor模式下,把开启Raycast Target选项的UI以蓝色线框的形式显示出来,方便大家检查遗漏的关闭的UI元素

Unity 优化之UGUI (2017年版【一】)_第17张图片

避免网格重建(Canvas.BuildBatch)

Unity 优化之UGUI (2017年版【一】)_第18张图片
Canvas.BuildBatch

现在untiy2017版本中Canvas.BuildBatch的主要耗时已经放到了多线程中

网格重建的意思是把Canvas下所有的ui合成一个Mesh,当有UI元素更改的时候就会重建这个Mesh,造成性能消耗。 采取的对应策略就是【动静分离】,在经常变动的UI元素(位置、颜色、图片等)上添加Canvas组件,就可以避免因为UI的改变造成整个Mesh全部重建。当然对于数量较多需要进行颜色渐变的UI元素都添加上canvas显然不合适,因为添加canvas会增加DrawCall。所以我们需要采用另一个策略,在Image上添加一个自定义的material,然后更改这个material的Tint属性,这样既能满足颜色渐变,又能避免网格频繁重建造成的性能消耗(实质是避免修改网格上的顶点属性,造成网格重建)

Unity 优化之UGUI (2017年版【一】)_第19张图片

Unity 优化之UGUI (2017年版【一】)_第20张图片

Unity 优化之UGUI (2017年版【一】)_第21张图片

额外注意的地点

  • 避免使用OutLine组件
  • 避免使用Shadow组件

应对策略:使用Text Mesh Pro插件是一个不错的选择(现在已经免费)

  • 避免频繁使UI元素SetActive(开、关),会造成网格重建

应对策略:使用CanvasGroup

你可能感兴趣的:(Unity 优化之UGUI (2017年版【一】))