http://blog.csdn.net/zhuyingqingfen/article/details/18979547

http://blog.csdn.net/lysc_forever/article/details/13614449

(1)什么是深度?

深度其实就是该像素点在3d世界中距离摄像机的距离。离摄像机越远,则深度值(Z值)越大。

(2)什么是深度缓存?

深度缓存中存储着准备要绘制在屏幕上的像素点的深度值。如果启用了深度缓冲区,在绘制每个像素之前,OpenGL会把该像素的深度值和深度缓存的深度值进行比较。如果新像素深度值<深度缓存深度值,则新像素值会取代原先的;反之,新像素值被遮挡,其颜色值和深度将被丢弃。(深度主要起的是比较的作用)

(3)什么是深度测试?

在深度测试中,默认情况是将要绘制的新像素的z值与深度缓冲区中对应位置的z值进行比较,如果比深度缓存中的值小,那么用新像素的颜色值更新深度缓存中对应像素的颜色值。

(4)为什么需要深度?

在不使用深度测试的时候,如果我们先绘制一个距离较近的物体,再绘制距离较远的物体,则距离远的物体因为后绘制,会把距离近的物体覆盖掉,这样的效果并不是我们所希望的。而有了深度缓冲以后,绘制物体的顺序就不那么重要了,都能按照远近(Z值)正常显示,这很关键。

(5)【颜色缓冲区】

  颜色缓冲区(COLOR_BUFFER)就是帧缓冲区(FRAME_BUFFER),你需要渲染的场景最终每一个像素都要写入该缓冲区,然后由它在渲染到屏幕上显示.

(6)【深度缓冲区

  深度缓冲区(DEPTH_BUFFER)与帧缓冲区对应,用于记录上面每个像素的深度值,通过深度缓冲区,我们可以进行深度测试,从而确定像素的遮挡关系,保证渲染正确.

(7)【模板缓冲区】

  模版缓冲(STENCIL_BUFFER)与深度缓冲大小相同,通过设置模版缓冲每个像素的值,我们可以指定在渲染的时候只渲染某些像素,从而可以达到一些特殊的效果.

那么,在unity中,如果知道了渲染队列,ZWrite,ZTest,如何确定哪个物体先显示呢?

首先,unity先将渲染队列中较前的进行渲染,然后再执行ZWrite,ZTest

ZWrite可以取的值为:On/Off,默认值为On,代表是否要将像素的深度写入深度缓存中(同时还要看ZTest是否通过)。

ZTest可以取的值为:Greater/GEqual/Less/LEqual/Equal/NotEqual/Always/Never/Off,默认值为LEqual,代表通过比较深度来更改颜色缓存的值。例如当取默认值的情况下,如果将要绘制的新像素的z值小于等于深度缓存中的值,则将用新像素的颜色值更新深度缓存中对应像素的颜色值。需要注意的是,当ZTest取值为Off时,表示的是关闭深度测试,等价于取值为Always,而不是Never!Always指的是直接将当前像素颜色(不是深度)写进颜色缓冲区中;而Never指的是不要将当前像素颜色写进颜色缓冲区中,相当于消失。

///

那么,重点来了:

1.当ZWrite为On时,ZTest通过时,该像素的深度才能成功写入深度缓存,同时因为ZTest通过了,该像素的颜色值也会写入颜色缓存。

2.当ZWrite为On时,ZTest不通过时,该像素的深度不能成功写入深度缓存,同时因为ZTest不通过,该像素的颜色值不会写入颜色缓存。

3.当ZWrite为Off时,ZTest通过时,该像素的深度不能成功写入深度缓存,同时因为ZTest通过了,该像素的颜色值会写入颜色缓存。

4.当ZWrite为Off时,ZTest不通过时,该像素的深度不能成功写入深度缓存,同时因为ZTest不通过,该像素的颜色值不会写入颜色缓存。

可以看到,像素的深度能否成功写入深度缓存,条件是ZWrite为On,ZTest通过;

写入深度缓存的作用就是为ZTest的比较做准备。

///

ZTest 和 ZWrite的注意点:

1.如果ZTest 测试通过,首先会将颜色写入对应的颜色缓冲区,然后再判断ZWrite是否开起,如果ZWrite Off 那么就不把像素的深度值写入深度缓冲区,如果ZWrite On 此时才把像素的深度值写入深度缓冲区。

2.如果ZTest 测试没有通过,首先不会将颜色写入颜色缓冲区,然后 不论ZWrite是否开启,都不会将像素的深度值写入深度缓冲区

///

因为ZWrite默认值为On,ZTest默认值为LEqual,所以这很好地解释了为什么在unity中,距离相机近的东西会阻挡住距离相机远的东西。如果我们先绘制一个距离较近的物体,再绘制距离较远的物体,则距离远的物体因为后绘制,会把距离近的物体覆盖掉,这时我们可以通过修改ZWrite和ZTest来改变物体的遮挡关系!

测试环境(蓝色方块距离相机较近,注意这个不是Game视图):

测试的Shader代码(两个方块的shader代码起始都是一样的,测试时修改的是测试区里的代码):

[csharp] view plain copy
  1. Shader "Custom/ZTest" {
  2. Properties {
  3. _MainTex ("Base (RGB)", 2D) = "white" {}
  4. }
  5. SubShader {
  6. Tags { "RenderType"="Opaque" }
  7. LOD 200
  8. /测试区
  9. Tags{ "Queue" = "Geometry" }
  10. //ZWrite Off
  11. //ZTest Off
  12. /测试区
  13. CGPROGRAM
  14. #pragma surface surf Lambert
  15. sampler2D _MainTex;
  16. struct Input {
  17. float2 uv_MainTex;
  18. };
  19. void surf (Input IN, inout SurfaceOutput o) {
  20. half4 c = tex2D (_MainTex, IN.uv_MainTex);
  21. o.Albedo = c.rgb;
  22. o.Alpha = c.a;
  23. }
  24. ENDCG
  25. }
  26. FallBack "Diffuse"
  27. }

渲染顺序:先蓝色方块再白色方块(以下简称蓝,白)

注意ZWrite默认值为On,ZTest默认值为LEqual,没有渲染物体时,深度缓存中的深度可以理解为无限大

1.

[csharp] view plain copy
  1. /蓝色方块测试区
  2. Tags{ "Queue" = "Geometry+200" }
  3. ZWrite Off
  4. //ZTest Off
  5. /蓝色方块测试区
[csharp] view plain copy
  1. /白色方块测试区
  2. Tags{ "Queue" = "Geometry+300" }
  3. //ZWrite Off
  4. //ZTest Off
  5. /白色方块测试区

结果:白在前面

分析:蓝没有将像素写进深度缓存中,ZTest通过了,颜色缓存中存放了蓝的颜色值

白将像素写进深度缓存中,ZTest通过了,颜色缓存的值变为白的,所以显示白

2.

[csharp] view plain copy
  1. /蓝色方块测试区
  2. Tags{ "Queue" = "Geometry+200" }
  3. //ZWrite Off
  4. //ZTest Off
  5. /蓝色方块测试区
[csharp] view plain copy
  1. /白色方块测试区
  2. Tags{ "Queue" = "Geometry+300" }
  3. //ZWrite Off
  4. //ZTest Off
  5. /白色方块测试区

结果:蓝在前面

分析:蓝将像素写进深度缓存中,ZTest通过了,颜色缓存中存放了蓝的颜色值

而白的像素深度大于蓝的,既没有成功将像素写进深度缓存,同时ZTest不通过,像素被舍弃,所以显示蓝

3.

[csharp] view plain copy
  1. /蓝色方块测试区
  2. Tags{ "Queue" = "Geometry+200" }
  3. ZWrite Off
  4. ZTest Off
  5. /蓝色方块测试区
[csharp] view plain copy
  1. /白色方块测试区
  2. Tags{ "Queue" = "Geometry+300" }
  3. //ZWrite Off
  4. //ZTest Off
  5. /白色方块测试区

结果:白在前面

分析:蓝没有将像素写进深度缓存中,ZTest通过了,颜色缓存中存放了蓝的颜色值

白将像素写进深度缓存中,ZTest通过了,颜色缓存中存放为白的颜色值

4.

[csharp] view plain copy
  1. /蓝色方块测试区
  2. Tags{ "Queue" = "Geometry+200" }
  3. //ZWrite Off
  4. //ZTest Off
  5. /蓝色方块测试区
[csharp] view plain copy
  1. /白色方块测试区
  2. Tags{ "Queue" = "Geometry+300" }
  3. //ZWrite Off
  4. ZTest Off
  5. /白色方块测试区

结果:白在前面

分析:蓝将像素写进了深度缓存中

白将像素写进了深度缓存中,ZTest通过了,白将颜色缓存中的蓝的像素颜色值替换了。

5.

[csharp] view plain copy
  1. /蓝色方块测试区
  2. Tags{ "Queue" = "Geometry+200" }
  3. //ZWrite Off
  4. //ZTest Off
  5. /蓝色方块测试区
[csharp] view plain copy
  1. /白色方块测试区
  2. Tags{ "Queue" = "Geometry+300" }
  3. ZWrite Off
  4. ZTest Off
  5. /白色方块测试区

结果:白在前面

分析:蓝将像素写进了深度缓存中

白没能将像素写进了深度缓存中(ZWrite为off),但ZTest通过了,此时颜色缓存的值变为白的,但是深度缓存的值是蓝的

6.

[csharp] view plain copy
  1. /蓝色方块测试区
  2. Tags{ "Queue" = "Geometry+200" }
  3. //ZWrite Off
  4. ZTest Off
  5. /蓝色方块测试区
[csharp] view plain copy
  1. /白色方块测试区
  2. Tags{ "Queue" = "Geometry+300" }
  3. //ZWrite Off
  4. ZTest Off
  5. /白色方块测试区

结果:白在前面

分析:蓝将像素写进了深度缓存中

白将像素写进了深度缓存中,ZTest通过了,此时颜色缓存的值变为白的

7.

[csharp] view plain copy
  1. /蓝色方块测试区
  2. Tags{ "Queue" = "Geometry+200" }
  3. ZWrite Off
  4. //ZTest Off
  5. /蓝色方块测试区
[csharp] view plain copy
  1. /白色方块测试区
  2. Tags{ "Queue" = "Geometry+300" }
  3. ZWrite Off
  4. //ZTest Off
  5. /白色方块测试区

结果:白在前面

分析:白、蓝都没有将像素写入深度缓存中,所以深度缓存中的深度值为无穷大,最后因为白中ZTest默认值的原因,所以显示白

8.

[csharp] view plain copy
  1. /蓝色方块测试区
  2. Tags{ "Queue" = "Geometry+200" }
  3. ZWrite Off
  4. ZTest Off
  5. /蓝色方块测试区
[csharp] view plain copy
  1. /白色方块测试区
  2. Tags{ "Queue" = "Geometry+300" }
  3. ZWrite Off
  4. ZTest Off
  5. /白色方块测试区

深度缓冲区,RenderQueue、ZWrite和ZTest相关推荐

  1. C++ opengl 深度缓冲区

    GL_DEPTH_BUFFER_BIT 每一次清除之前,都要清空深度缓冲区 启动深度测试glEnable(GL_DEPTH_TEST):离视野近的物体会挡住离视野远的物体. 运行截图如下: 源码如下: ...

  2. Unity Shader 阴影系列(2)--内置阴影源码解析

    Unity是如何生成阴影的 前言 相关的宏 投射阴影 v2f结构定义 V2F_SHADOW_CASTER UNITY_VERTEX_OUTPUT_STEREO 顶点函数:TRANSFER_SHADOW ...

  3. 图形学进阶——深度与模板测试

    Z Test & Stencil Test深度测试与模板测试 百人计划学习地址:图形 3.1 深度与模板测试 传送门效果示例 一.模板测试 Stencil Test 流程: 1. 什么是模板测 ...

  4. Unity半透明特效原理讲解(为什么半透明设置渲染顺序和深度写入这么重要)

    Unity半透明特效原理讲解(为什么半透明设置渲染顺序和深度写入这么重要 写在前面 实验场景 实验1:红(不透明)+蓝(不透明)+默认渲染顺序(先渲染蓝Cube) 实验2:红(不透明+优先渲染)+蓝( ...

  5. Unity Shader PostProcessing - 11 - Depth Fog/Height Fog - 雾效/深度雾效/高度雾/深度+高度雾

    文章目录 观察生活中的雾 Scenario 1 - Pure Depth Fog - 纯深度雾 提取有用信息 Scenario 2 - Height Fog - 带高度的雾 提取有用信息 Scenar ...

  6. Unity Shader-边缘检测效果(基于颜色,基于深度法线,边缘流光效果,转场效果)

    前言 周末通关了一个小游戏,流程很短,6个小时左右就通关,但是游戏的画风,视角,玩法都比较新奇,对了,游戏的名字也很奇特<12 Is Better Than 6>(12比6好是有什么梗吗? ...

  7. ARCore深度渲染问题分析

    文章目录 1.前言 2.深度图显示 3.深度遮挡 3.1 处理流程 3.2 相关代码 4.结语 1.前言 ARCore深度效果显示分为两部分:第一部分是深度图显示,另一部分为深度遮挡(即实现真实物体与 ...

  8. Unity 基于法线和深度实现完美描边,可独立控制物体描边

    目录 前言 自定义PostProcess OutlineShader关键代码说明 1 使用深度绘制描边 1.1 获得斜四方形UV坐标: 1.2 采样四方向深度 2 使用法线绘制描边 3 解决倾斜表面白 ...

  9. Unity3D 阴影和深度纹理总结

    首先介绍一下,阴影的生成,阴影一般是在屏幕空间生成的,利用Unity3D引擎实现阴影的绘制,首先要明白其实现原理,Unity5的Demo已经为我们提供了代码实现,先介绍一下阴影的视线原理: 屏幕空间的 ...

最新文章

  1. 二进制地址的伙伴地址
  2. 爬虫综合大作业(震惊!爬取了590位微信好友后竟然发现了)
  3. Linux的一些配置文件位置
  4. 关于 jMeter 结果报表里的 APDEX (Application Performance Index)
  5. python getopt参数参数自动补全_如何在Python中使用getopt / OPTARG?如果给出过多的参数(9),如何转移参数?...
  6. 安卓团课快进_青年大学习网上主题团课第十季第七期答案
  7. 读这样的文章才能清楚什么是RIA
  8. js后退一直停留在当前页面或者禁止后退
  9. oracle脱敏脚本
  10. 廖雪峰git教程学习记录
  11. Nmap小技巧——探测大网络空间、局域网中的存活主机
  12. 腾讯云租用CentOS 7.2 64
  13. UVA 10066 10192
  14. Unittest-单元测试2
  15. 仿网易云音乐的小程序项目(粗糙版)
  16. 搭建Discuz论坛网站-最新版Discuz3.4
  17. 计算机在生活中的应用论文2000字,浅议计算机在生活中的应用论文(2)
  18. 15年计算机考研大纲,2015计算机考研大纲
  19. 前端学习 day4 : 盒子模型,浮动,定位
  20. cent os 7 与cent os 6区别

热门文章

  1. nginx: [emerg] bind() to 0.0.0.0:66 failed (98: Address already in use)
  2. 【PostgreSQL-9.6.3】extract函数
  3. [Unity3D]无缝场景切换解决方案(1) - 简单场景切换
  4. 【异常:Could not resolve】react-native run-android
  5. 解决CentOS无法解析域名的问题
  6. PlSqlDev中执行INSERT SQL语句包含符号导致数据异常
  7. 安装pytest时遇到的问题及解决方案
  8. AppScan漏洞“已解密的登陆请求”修复解决方案
  9. 项目出现 The superclass “javax.servlet.http.HttpServlet“ was not found on the Java Build Path 解决方法
  10. Missing artifact com.oracle:ojdbc6:jar:11.2.0.3 Maven中不能引入ojdbc解决方法,错误