可以通过Frame debugger查看每个drawcall绘制了哪些东西

UGUI源码下载地址:https://bitbucket.org/Unity-Technologies/ui/downloads/

本文测试环境:unity2018.2.9f1,基于Unity Editor (PC平台)

合批的过程

网格更新机制

  • Cavans.SendWillRenderCanvas

    • m_LayoutRebuildQueue
    • m_GraphicRebuildQueue
  • Canvas.BuildBatch 更新所有DrawCall

    • WaitingForJob 子线程网格合并
    • PutGeometryJobFence
    • BatchRendere.Flush UI如果开多线程渲染,BatChRender.Flush会增高,主线程在等待子线程的结果时Flush会等待。

哪些因素的改变会引起合批

从源码中可以看到,这些数据的改变会引起合批

源码地址: UI / UnityEngine.UI / UI / Core / Utility / VertexHelper.cs

private List<Vector3> m_Positions = ListPool<Vector3>.Get();//顶点位置的拷贝或指定新顶点位置的数组
private List<Color32> m_Colors = ListPool<Color32>.Get();//颜色
private List<Vector2> m_Uv0S = ListPool<Vector2>.Get();//基本纹理坐标
private List<Vector2> m_Uv1S = ListPool<Vector2>.Get();//第二套纹理坐标
private List<Vector2> m_Uv2S = ListPool<Vector2>.Get();//第三套纹理坐标
private List<Vector2> m_Uv3S = ListPool<Vector2>.Get();
private List<Vector3> m_Normals = ListPool<Vector3>.Get();//法线
private List<Vector4> m_Tangents = ListPool<Vector4>.Get();//切线
private List<int> m_Indices = ListPool<int>.Get();//mesh的索引

Mesh的API:http://wiki.ceeger.com/script/unityengine/classes/mesh/mesh

http://wiki.ceeger.com/script/unityengine/classes/mesh/mesh.getindices

怎么避免合批

尽量减少“动态”长文本(运行时修改文本内容)

Image或Text,如果不需要点击,则不要勾选Raycasts

降低界面的更新频率

避免图集分离,使用相同的图集。

同一图集的Image元素应尽量保证在Hierarchy中连续,避免中间插入其他图集,或插入文本。

避免图片叠加在一起(遮挡,旋转)

如果sprite是中心镂空且切图为九宫格时,可以去除fill center,以减少over draw

透明Image,用来做响应点击事件,同样存在开销

避免或减少Mask的使用,1个Mask至少增加两个DC

避免频繁删除/增加UI对象,UI层次结构变化会引起Canvas的更新

避免频繁动态的更新UI元素的Vertex, Rect, Color, Material, Texture等,可能引起Canvas数据更新和Batch更新计算,有可能引起VBO Update(重新提交顶点数据)。

尽可能使用少的UI Material和贴图(使用图集),使得可以Batching。

同一父节点下所有子节点,保持相同的层次结构(如List控件下的item),便于底层相同depth下UI元素Batch。

避免UI元素数目过多和层次结构过于复杂影响Batch更新速度。

固定的Text考虑与背景图层合在一张图上(可能不便本地化,但可以减少drawcall)。

使用缓存池,对缓存频繁使用的元素。

部分内容参考:http://gad.qq.com/article/detail/25947

HUD处理(动静分离)

Canvas重建就是为了合并DC,将经常变化的文字放在独立的Canvas,手动分离Canvas(会增加DC,不能和其它文字合并),但文字变化时其它Canvas就不需要重建。

示例:名字和血条分开在两个不同的节点下。这样当血条变化时,就不会引起名字的更新。如下图所示:

2、设置scale为0,而不是设置active = false/true,或者添加Alpha Group,设置alpha=0/1

不勾选FillCenter

镂空九宫格不勾选FillCenter,在Scene的Overdraw下可以查看到,不勾选FillCenter,overdraw会减少。

少用Effect功能

少用Outline,Tiled Sprite

outline额外生成7倍顶点

在一个空场景中,给Text添加outline之后,顶点数大约是未添加之前的7.5倍。

去掉outline之后,顶点数下降了很多。

Image不使用Tiled

type=simple时的顶点数

使用Tiled之后,顶点数也上涨很多。

参考资料

Unity官方论坛发布 Unity UI性能优化技巧

UGUI优化:批次合并源码分析及工具

工具:UI层级辅助工具,用于显示UI的层级、批次等数据,便于UI性能优化。使用者可以结合以上规则,分析当前UI元素排列顺序、材质贴图设置,优化UI Batching,减少UI Drawcall。

Unity UI优化小结

重建 是UGUI优化的关键 -- Unite2017嘉宾杨怀忠分享《UGUI深度优化》

UGUI合批原理笔记的更多相关文章

  1. 多线程之CountDownLatch的用法及原理笔记

    前言-CountDownLatch是什么? CountDownLatch是具有synchronized机制的一个工具,目的是让一个或者多个线程等待,直到其他线程的一系列操作完成. CountDownL ...

  2. 磁盘文件I/O,SSD结构,局部性原理 笔记

    磁盘文件I/O过程 进程向内核发起read scene.dat请求: 内核根据inode获取对应该进程的address space,在address space查找page_cache,如果没有找到, ...

  3. Http协议工作特点和工作原理笔记

    工作特点: (1)B/S结构(Browser/Server,浏览器/服务器模式) (2)无状态 (3)简单快速.可使用超文本传输协议.灵活运行传输各种类型 工作原理: 客户端发送请求浏览器 -> ...

  4. vuex原理笔记

    本文总结自: https://tech.meituan.com/vuex-code-analysis.html, 将要点提炼为笔记,以便不时之需,安不忘危. 核心可分为两部分: 1.vue.use(V ...

  5. MOOC 编译原理笔记(一):编译原理概述以及程序设计语言的定义

    编译原理概述 什么是编译程序 编译程序指:把某一种高级语言程序等价地转换成另一张低级语言程序(如汇编语言或机器代码)的程序. 高级语言程序-翻译->机器语言程序-运行->结果. 其中编译程 ...

  6. 机器学习之主成分分析PCA原理笔记

    1.    相关背景 在许多领域的研究与应用中,通常需要对含有多个变量的数据进行观测,收集大量数据后进行分析寻找规律.多变量大数据集无疑会为研究和应用提供丰富的信息,但是也在一定程度上增加了数据采集的 ...

  7. 学习Elasticsearch原理笔记

    Elasticsearch是一个分布式可拓展的实时搜索和分析引擎 分布式实时文件存储,并将每一个字段都编入索引,使其可以被搜索 实时分析的分布式搜索引擎 可以拓展到上百台服务器,处理PB级别的结构化或 ...

  8. Kafka原理笔记

    1.什么是kafka? Kafka是一种分布式的,基于发布/订阅的消息系统(消息队列). 2.为什么要用kafka? 当业务逻辑变得复杂,数据量也会越来越多.此时可能需要增加多条数据线,每条数据线将收 ...

  9. BootStrap栅格系统原理 笔记

    1.内容居中:效果 关键代码: <div class="container"> .........之前上面添加在body标签下的代码 </div>添加cla ...

随机推荐

  1. Python - Python2与Python3的区别、转换与兼容

    区别 Python2.x与Python3.x版本区别:http://www.runoob.com/python/python-2x-3x.html 示例解读Python2和Python3之间的主要差异 ...

  2. 【译】编写支持SSR的通用组件指南

    原文来自:https://blog.lichter.io/posts/the-guide-to-write-universal-ssr-ready-vue-compon?utm_campaign=Vu ...

  3. Python内置函数(23)——format

    英文文档: format(value[, format_spec]) Convert a value to a “formatted” representation, as controlled by ...

  4. Python内置函数(25)——getattr

    英文文档: getattr(object, name[, default]) Return the value of the named attribute of object. name must ...

  5. SpringBoot+Elasticsearch

    1.  前言 1.1.  集成方式 Spring Boot中集成Elasticsearch有4种方式: REST Client Jest Spring Data Spring Data Elastic ...

  6. Android--SurfaceView播放视频

    前言 本篇博客讲解一下如何在Android下,使用SurfaceView播放一个视频流媒体.之前有讲到如何使用MediaPlayer播放音频流媒体,其实MediaPlayer还可以播放视频,只需需要S ...

  7. JAVA实现接口监控报警系统

    公司一内部系统需要添加一个接口和定时任务监控机制,针对了系统出现的定时任务没有执行,定时任务执行异常出错,对外传送的数据接口异常出错,对内的系统数据互传异常出错等问题进行设计,目的是为了能够让用户没发 ...

  8. 初探设计模式5:Spring涉及到的9种设计模式

    设计模式作为工作学习中的枕边书,却时常处于勤说不用的尴尬境地,也不是我们时常忘记,只是一直没有记忆. 今天,螃蟹在IT学习者网站就设计模式的内在价值做一番探讨,并以spring为例进行讲解,只有领略了 ...

  9. Java的类加载器种类(双亲委派)

    Java类加载器采用双亲委派模型: 1.启动类加载器:这个类加载器负责放在<JAVA_HOME>\lib目录中的,或者被-Xbootclasspath参数所指定的路径中的,并且是虚拟机识别 ...

  10. IDEA使用总结

    IDEA常用设置 在我们第一眼看见IDEA是这个样子的: 显示工具条 我们要显示工具条!,两个按钮哦 黑色主体 我们要黑色的主题,白色的太low了! 调整字体大小 现在的字体太小了,我要鼠标滑轮+cr ...