[转] Draw Call未被批处理?告诉你在Unity 5.6中如何查找原因 [复制链接]
Unity在5.6之前的版本中并未提供很直接的方式来查找Draw Call未被批处理的原因,但Unity 5.6在Frame Debugger中新增了一项功能,帮助开发者查找相关信息。今天这篇文章就为大家分享,在Unity 5.6中如何查看Draw Call未被批处理的原因。
相信大家都知道,Unity内置的动态与静态批处理有助于减少游戏中的Draw Call数量。在Stats窗口中,当“Saved by batching“值大于零时就表示批处理已经生效。但不幸的是,要想知道批处理为何不生效却比较困难,虽然Unity手册提供了一些可能的原因,但这些信息需要用户已经了解相关的基础知识。
好消息是,Unity 5.6的Frame Debugger中加入了一个新的功能,用于告知我们Unity发起新的批处理的原因。
Frame Debugger是Unity 5.x推出的新功能,依次点击菜单项Window > Frame Debugger可以打开Frame Debugger窗口。该窗口显示游戏中所有的批处理,以及这些批处理所有的细节信息,包括着色器、纹理及批处理所用的大量设置信息等。
在详细介绍Unity何时发起新的批处理之前,先来了解批处理的概念及其作用。
Unity 5.6中的Frame Debugger现解释了Unity发起新批处理的原因。
何为批处理(Batch)
Unity为了在屏幕上绘制游戏对象,它需要向图形API发起一次“绘制”命令。该过程其实就是一次“Draw Call”。但在发起命令之前,Unity还需要为绘制的对象设置所需的GPU状态:网格、着色器、纹理、混合设置以及一些其他的着色器属性。而状态改变命令再加上一个或多个绘制命令就称为一次批处理。
批处理过程(Batching)
导致批处理缓慢的原因就是改变GPU状态的命令,而绘制命令实际上仅占用很少的资源。所以Unity总是试图利用同一个GPU状态同时渲染多个对象。这一过程被称为批处理。
Unity提供三种类型的批处理:静态批处理,动态批处理以及GPU Instancing。
- 静态批处理会在构建时将多个静态网格对象合并为一个或多个大的网格对象,然后在运行时一次批处理渲染一个大网格中的多个对象。
- 动态批处理在每帧中获取多个小型网格对象,在CPU中对其进行顶点变换,将相似的顶点组合到一起,然后一次性绘制它们。
- GPU Instancing(Unity 5.4中引入)可以利用少量Draw Call绘制多个具有不同的位置、旋转以及其他着色器属性的相同对象。
导致批处理失败的原因
有时在编辑器中可以清楚地看到,一些本应被批处理的对象出于某些原因没有被批处理。首先,请检查Player Settings中是否启用批处理功能。这个步骤看似多余,但我们遇到太多的失败案例都是因为忘记开启该功能。
我们专门为此提供了示例工程,用来演示Unity在什么情况下必须发起新的批处理请求。首先下载该工程并导入Unity项目。请注意,您需要使用Unity 5.6才能看到Frame Debugger中关于批处理状态的说明。
以下是示例工程(Unity 5.6)中导致无法进行批处理的原因。每个原因对应一次单独的批处理。
- Additional Vertex Streams — 对象使用MeshRenderer.additionalVertexStreams设置了额外的顶点信息流。
- Deferred Objects on Different Lighting Layers — 该对象位于另一不同的光照层中。
- Deferred Objects Split by Shadow Distance — 两个物体中有一个在阴影距离范围内而另一个不是。
- Different Combined Meshes — 该对象属于另一个已合并的静态网格。
- Different Custom Properties — 该对象设置了不同的MaterialProperyBlock。
- Different Lights — 该对象受不同的前向光照(Forward Light)影响。
- Different Materials — 该对象使用不同的材质。
- Different Reflection Probes — 该对象受不同的反射探头(Reflection Probe)影响。
- Different Shadow Caster Hash — 该对象使用其他的阴影投射着色器,或是设置了不同的着色器参数/关键字,而这些参数/关键字会影响阴影投射Pass的输出。
- Different Shadow Receiving Settings — 该对象设置了不同的“Receive Shadows”参数,或是一些对象在阴影距离内,而另一些在距离之外。
- Different Static Batching Flags — 该对象使用不同的静态批处理设置。
- Dynamic Batching Disabled to Avoid Z-Fighting — Player Settings中关闭了动态批处理,或在当前环境中为避免深度冲突而被临时关闭。
- Instancing Different Geometries — 使用GPU Instancing渲染不同的网格或子网格。
- Lightmapped Objects — 对象使用了不同的光照贴图,或在相同的光照贴图中有不同的光照贴图UV转换关系。
- Lightprobe Affected Objects — 对象受其他光照探头(Light Probe)影响。
- Mixed Sided Mode Shadow Casters — 对象的“Cast Shadows”设置不同。
- Multipass — 对象使用了带多个Pass的着色器。
- Multiple Forward Lights — 该对象受多个前向光渲染影响。
- Non-instanceable Property Set — 为instanced着色器设置来non-instanced属性。
- Odd Negative Scaling — 该对象的缩放存在某个奇数为负值,例如(1,-1,1)。
- Shader Disables Batching — 着色器使用“DisableBatching”标签显式关闭了批处理。
- Too Many Indices in Dynamic Batch — 动态批处理索引过多(超过32k)。
- Too Many Indices in Static Batch — 静态批处理中的组合网格索引过多。对于OpenGL ES来说是48k,OSX是32k,其他平台是64k。
- Too Many Vertex Attributes for Dynamic Batching — 欲进行动态批处理的子网格拥有超过900个顶点属性。
- Too Many Vertices for Dynamic Batching — 欲进行动态批处理的子网格顶点数量超过300个。
结论
现在可以开始使用Frame Debugger新功能来检查工程了,帮助您找到可以进行批处理的内容。随着引擎的不断更新,将来可能会添加更多导致批处理失败的原因,请关注我们在Github上的项目。
[转] Draw Call未被批处理?告诉你在Unity 5.6中如何查找原因 [复制链接]的更多相关文章
- ORA-01219:数据库未打开:仅允许在固定表/视图中查询
好久没有登陆到Oracle的服务器了,把密码都忘记了.sql>conn sys/sys as sysdba;sql>alter user system identified by *;结果 ...
- 通过批处理操作注册表实现winform应用中Webbrowser以指定的IE版本加载网页
通过批处理操作注册表实现winform应用中Webbrowser以指定的IE版本加载网页 rem 强制WebBrowser控件使用指定IE版本显示应用的网页 IF EXIST %windir%\Sys ...
- SqlServer 2014该日志未截断,因为其开始处的记录是挂起的复制操作或变更数据捕获
环境:AlwaysOn集群 操作系统:Windows Server 2008 R2 数据库: SQL Server 2014 错误提示:“该日志未截断,因为其开始处的记录是挂起的复制操作或变更数据捕获 ...
- 批处理at命令--一切尽在计划中
让计算机在自己规定的时间里干自己规定的事,一切尽在计划之中.所以at命令你一定不能错过. 概述 列出在指定的时间和日期在计算机上运行的已计划命令或计划命令和程序,以及设置在指定时间和日期在计算机上运行 ...
- 【Silverlight】打开Silverlight程序报错,"未找到导入的项目......请确认<Import>声明中的路径正确,且磁盘上存在该文件"
在打开Silverlight程序时,报错(如图所示),程序使用的是Visual Studio 2013和最新的Silverlight版本(Silverlight5). 然后我在网上找了下说:Silve ...
- Button未设type属性时在非IE6/7中具有submit特性
代码如下 <!DOCTYPE html> <html> <head> <title>Button在Form中具有submit的特性</title& ...
- Java学习----你可以告诉对象该怎么做(方法中传参)
对象根据参数传递来的条件执行相应的功能. package org.demo.app2; public class App2 { public void print(String msg, int nu ...
- 使用layui框架 修改时部分参数未传给后台(查找原因)
采用的结构: <form class="layui-form reset-form" action="" id="formData"& ...
- 【Azure 批处理 Azure Batch】在Azure Batch中如何通过开始任务自动安装第三方依赖的一些软件(Windows环境)
准备条件 Azure Batch账号 需要安装的软件包(zip)文件,里面包含该软件的msi安装文件, 此处使用python.msi 版本 3.3.3 作为例子(https://www.python. ...
随机推荐
- 使用airmon-ng工具开启监听模式
使用ifconfig命令查看活动的网络接口 可以看出网卡已经激活了,然后将网卡设置为混杂模式 root@sch01ar:~# airmon-ng start wlan0 用ifconfig查看网卡是否 ...
- 使用Java进行远程方法调用的几个方案及比较
Java远程方法调用是编程过程中比较常见的问题,列举一下主要包括如下几类: 1.Java RMI (Remote Method Invocation) 2.EJB远程接口调用 3.WebService ...
- OSGI 模块化
推荐教程:https://course.tianmaying.com/osgi-toturial+osgi-concept#15
- Listview使用安卓自带布局实现单选
安卓提供了一些自带的布局,使用非常简单.直接看代码吧. package com.example.linfeng.myapplication; import android.app.Activity; ...
- MongoDB安全加固方案,防止数据泄露被勒索
早上起来,发现生产数据库被删了,留下一个数据库名叫“PLEASE_READ”,里面内容如下: "Info" : "Your DB is Backed up at our ...
- bash shell笔记1 脚本基础知识
原创作品,允许转载,转载时请务必以超链接形式标明文章 原始出处 .作者信息和本声明.否则将追究法律责任.http://twentyfour.blog.51cto.com/945260/505644 * ...
- grep家族
grep家族由命令grep.egrep和fgrep组成. grep:在文件中全局查找指定的正则表达式,并且打印所有包含该表达式的行.egrep和fgrep是grep的变体.egrep:grep的扩展, ...
- google/dense_hash_map
这个库使用时需要注意的地方: 1.在插入数据之前,需要先调用set_empty_key()设置一个空Key,Key的值可以为任意符合类型的.但请注意之后插入的Key不能和空Key相同,否则会abort ...
- 【转】Provisional headers are shown
在chrome开发者工具的 Network 面板中,某些请求头后面会跟着下面这行文字: Provisional headers are shown 这种请求实际上根本没有产生,对应的请求头当然也不应该 ...
- 在Ubuntu16.04上使用rz上传文件,XXX was skipped
原本想把hadoop-2.8.5.tar.gz上传到/usr/local/src文件夹下,报错,was skipped 如下图: 换个文件夹位置,更换到本用户文件夹下,可以上传,说明是对文件夹操作权限 ...