原文连接:https://www.cnblogs.com/ljx12138/p/5341381.html

参考另一篇写的比较好的:Unity Shader中的 ZTest & ZWrite

初学Shader,一开始对于渲染队列,ZTest 和 ZWrite一头雾水,经过多方查阅和实验,有了一些自己的理解。发此文与初学Shader的朋友分享,也算是为自己做个笔记。不对或不足之处欢迎指正。

不说废话,直接进入正题。

首先是实验场景,一蓝一红两个Cube。蓝在红前。

两个方块所使用的Shader都是最简单的 V&F 着色程序,不同的是蓝色方块alpha返回值为0.6,红色为1。

但是蓝色方块并没有丝毫透明的效果,这时我们在蓝色方块的Shader内加上这样一行代码 :

Blend SrcAlpha OneMinusSrcAlpha 。

 

稍微解释一下,这行指令意思就是将本 Shader 计算出的颜色值(源颜色值,即蓝色) * 源Alpha值(0.6) + 目标颜色值(可以理解为背景色) * (1-0.6),从而让蓝色方块展示出了40%的透明度。

然后我们看到了这样的效果:

似乎有些透明的样子了,但是红色方块还是显示不出来。我们再加上这样一行代码:

Tags {"Queue" = "Transparent"}

 

意思是设置它在渲染队列中的值为 Transparent (透明) = 3000,值越小越先渲染,而后渲染( Queue 值大)的物体会覆盖先渲染的物体(红块未设置 Queue 值,默认为 Geometry(几何体) = 2000)。在理想的世界中,我们应该让远处的物体先渲染,近处的物体后渲染,这样远处的物体就不会遮挡住近的物体。

接下来我们看到了正确的结果:

Queue 其他预定义的值为:Background = 1000 , AlphaTest = 2450,Overlay = 4000。默认值是Geometry 。

例如:如果我们想让远处的红色方块遮挡住近处的蓝色方块,即让蓝色的先渲染,红色的后渲染,只需将红块的Queue 也设置为 Transparent ,蓝块的Queue值-1。

然后我们看看效果:

好像并没有什么卵用。。。这是为什么呢?

原因在于这样两条指令:

虽然我们的代码里并没有这两行,但它们是默认存在的。

ZTest ,深度测试;LEqual ,小于等于。

ZWrite ,深度写入,On ,打开。

ZTest 可取值为:Greater , GEqual , Less , LEqual , Equal , NotEqual , Always , Never , Off,默认是 LEqual,ZTest Off 等同于 ZTest Always。

ZWrite 可取值为:On , Off,默认是 On。

系统中存在一个颜色缓冲区和一个深度缓冲区,分别存储颜色值和深度值,来决定画面上应该显示什么颜色。

深度值是物体在世界空间中距离摄像机的远近。距离越近,深度值越小;距离越远,深度值越大。

例如在我们的场景中,蓝色方块比红色更靠近相机,蓝块的深度值就比红块小。

假设蓝块的深度值为 0.5,红块为 0.7。还记得在上面的例子中,我们让蓝块在渲染队列中排在红块前面,系统就先将蓝色值存入了颜色缓冲区中对应的区域,将深度值 0.5 存入了深度缓冲区中对应的区域。接下来渲染红块,系统会将红块的深度值与深度缓冲区中的值进行比较(这个过程就是深度测试),由于默认的 ZTest 深度测试的方式是 LEqual 小于等于,即深度值小于等于 0.5 的颜色才会通过测试。如果通过了测试,且 ZWirte 处于 On 的状态,该颜色的深度值就会替代深度缓冲区中的值,颜色值也会替代颜色缓冲区中的值,从而显示出新颜色。

很显然,0.7 > 0.5,所以红色并不能通过测试,红块也就不能显示在蓝块前面。

如果我们硬要远处的红块遮挡住近处的蓝块,很显然,我们应该改变或关闭深度测试,或者关闭深度写入(关闭了深度测试或者深度写入之后,物体颜色的遮挡关系就会和渲染队列一致,即排在后面的会挡住前面的)。

接下来我们试试关闭蓝块的深度写入:

得到了想要的结果:

再试试单独关闭红块的深度测试,注释蓝块的深度写入:

 

结果还是一样。但如果我们改变红块深度测试的方式呢:

即深度值大于深度缓冲区中的值就能通过测试,还记得我们假设红块深度值为 0.7,蓝块为 0.5。理论上会得到我们想要的结果:

奇怪的是,红块的另一半去哪了?

答案是被背景挡住了。

按照距离相机的远近,我们可以假设背景的深度值为 1。在消失的另一半的深度缓冲区中的深度值应该是背景的深度值 1。而我们设置了 ZTest Greater,0.7 < 1,所以红块另一半没有通过深度测试。

【转】Unity ZTest 深度测试 & ZWrite 深度写入的更多相关文章

  1. Unity ZTest 深度测试 & ZWrite 深度写入

    初学Shader,一开始对于渲染队列,ZTest 和 ZWrite一头雾水,经过多方查阅和实验,有了一些自己的理解.发此文与初学Shader的朋友分享,也算是为自己做个笔记.不对或不足之处欢迎指正. ...

  2. Unity ZTest深度测试 & ZWrite深度写入

    Shader深度渲染队列Queue预定义值:Background(1000).Geometry(2000).AlphaTest(2450).Transparent(3000).Overlay(4000 ...

  3. Unity Shader-渲染队列,ZTest,ZWrite,Early-Z

    在渲染阶段,引擎所做的工作是把所有场景中的对象按照一定的策略(顺序)进行渲染.最早的是画家算法,顾名思义,就是像画家画画一样,先画后面的物体,如果前面还有物体,那么就用前面的物体把物体覆盖掉,不过这种 ...

  4. [UnityShader基础]02.深度测试 & 深度写入

    参考链接: https://blog.csdn.net/v_xchen_v/article/details/79380222 前面说到了渲染队列,对于两个不透明的物体A和B,它们处于同一个渲染队列中. ...

  5. Unity编辑器 - 资源修改立即写入磁盘AssetDataBase.SaveAssets()

    Unity编辑器 - 资源修改立即写入磁盘AssetDataBase.SaveAssets() 在编写编辑器时,如果需要修改Unity序列化资源(如Prefab,美术资源,ScriptableObje ...

  6. Unity加载模块深度解析(Shader)

    作者:张鑫链接:https://zhuanlan.zhihu.com/p/21949663来源:知乎著作权归作者所有.商业转载请联系作者获得授权,非商业转载请注明出处. 接上一篇 加载模块深度解析(二 ...

  7. Unity加载模块深度解析(网格篇)

    在上一篇 加载模块深度解析(一)中,我们重点讨论了纹理资源的加载性能.这次,我们再来为你揭开其他主流资源的加载效率. 这是侑虎科技第53篇原创文章,欢迎转发分享,未经作者授权请勿转载.同时如果您有任何 ...

  8. Unity加载模块深度解析(纹理篇)

    在游戏和VR项目的研发过程中,加载模块所带来的效率开销和内存占用(即“加载效率”.“场景切换速度”等)经常是开发团队非常头疼的问题,它不仅包括资源的加载耗时,同时也包含场景物件的实例化和资源卸载等.在 ...

  9. Unity --- 如何降低UI的填充率

    1.首先简单介绍一下什么叫填充率: Fill Rate(填充率)是指显卡每帧或者说每秒能够渲染的像素数.在每帧绘制中,如果一个像素被反复绘制的次数越多,那么它占用的资源也必然更多.目前在移动设备上,F ...

随机推荐

  1. Zookeeper的客户端使用

    1.1 Zookeeper API(原生) 1)连接的创建是异步的,需要开发人员自行编码实现等待 2)连接没有超时自动的重连机制 3)Zookeeper本身没提供序列化机制,需要开发人员自行指定,从而 ...

  2. 新建swap分区

    1.在一块新盘上创建一个主分区,大小为1G大小. 2.将该硬盘数据变更为82(swap),并进行保存 3.查看是否已经将新建分区更改成了swap分区 4.将/dev/sdb2的标签设置为swap-sd ...

  3. CH5105 Cookies[线性DP]

    http://contest-hunter.org:83/contest/0x50%E3%80%8C%E5%8A%A8%E6%80%81%E8%A7%84%E5%88%92%E3%80%8D%E4%B ...

  4. cookie的使用以及cookie的跨域名获取

    cookie存放容量4k左右,可设置过期时间. 1.cookie的封装使用 //设置cookies function setCookie(name, value) { var Days = 30; v ...

  5. Acwing-282-石子合并(区间DP)

    链接: https://www.acwing.com/problem/content/284/ 题意: 设有N堆石子排成一排,其编号为1,2,3,-,N. 每堆石子有一定的质量,可以用一个整数来描述, ...

  6. 使用es6新增Set函数快速数组去重

    使用new Set()快速数组去重: let arr = [1, 2, 2, 3, 4, 5, 5, 5, 6] let set = new Set([...arr]) console.log([.. ...

  7. 【leetcode】1257. Smallest Common Region

    题目如下: You are given some lists of regions where the first region of each list includes all other reg ...

  8. dispatchEvent 自定义触发事件,常用于自定义鼠标事件或点击事件

    自定义事件的触发又是不可避免的,由于浏览器兼容性问题,我们要分开说了,针对标准浏览器和IE6/7等考古浏览器. 1. 对于标准浏览器,其提供了可供元素触发的方法:element.dispatchEve ...

  9. ubuntu1804 安装Nvidia驱动-nvidia-docker

    操作 看显卡 lspci | grep N 01:00.0 3D controller: NVIDIA Corporation GM107M [GeForce GTX 960M] (rev a2) 0 ...

  10. JavaScript RegExp ——对象,语法,修饰符,方括号,元字符,量词,对象方法,对象属性

    ㈠RegExp 对象 正则表达式是描述字符模式的对象. 正则表达式用于对字符串模式匹配及检索替换,是对字符串执行模式匹配的强大工具. ㈡语法 var patt=new RegExp(pattern,m ...