专栏首页进击的多媒体开发OpenGL 深度测试与精度值的那些事

OpenGL 深度测试与精度值的那些事

在 OpenGL 世界里,使用深度测试可以来防止被阻挡的面渲染到其他面的前面。

直接看一个没有使用深度测试的绘制:

未开启深度测试的情况

按照计划是绘制一个封闭的立方体,六个面都是有的,可从上面的效果来看并不是,立方体的有些面丢失了,只有后面的那个面,前面的面没了。

这就是在没有开启深度测试的情况下,本来应该被遮挡的,绘制在后面的面却绘制到了其他面之上。

要解决这种问题,就得使用深度测试了。

值得一提的是:在没有开启深度测试的情况下,假设绘制了多个不同远近的物体,那么对于最后的景象来说,哪怕是距离最远的,只要它的最后绘制的,都会显示在景象的前面。

当深度测试被启用时,OpenGL 会将一个片段的深度值与深度缓冲的内容进行对比。OpenGL 会执行一个深度测试,如果这个测试通过了的话,深度缓冲将会更新为新的深度值,如果深度测试失败了,该片段将会被丢弃。

深度缓冲是在片段着色器运行之后,在屏幕空间中运行的。屏幕空间坐标与通过 OpenGL 的 glViewport 所定义的视口密切相关,并且可以通过 GLSL 的内建变量 gl_FragCoord 从片段着色器中直接访问。

gl_FragCoord 的 x 和 y 分量代表了片段的屏幕空间坐标(其(0,0)位于左下角)。gl_FragCoord 中也包含了一个 z 分量,它包含了片段真正的深度值。z 值就是需要与深度缓冲内容所对比的那个汁。

深度缓冲默认是禁止的,通过如下代码开启它:

1glEnable(GL_DEPTH_TEST);

开启之后,如果一个片段通过了深度测试的话,OpenGL 就会在深度缓冲中存储该片段的 z 值;如果没有通过深度缓冲,则丢弃该片段。

如果开启了深度缓冲,就应该在每个渲染迭代之前,也就是 onDrawFrame 方法中清除深度缓冲,否则就仍在使用上一次渲染迭代时写入的深度值。

1// 清除深度缓冲
2glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);

如果在某些情况下,需要对所有的片段执行深度测试并丢弃某些片段,但是又不希望深度缓冲被更新,这时就需要使用一个只读深度缓冲

OpenGL 运行我们禁用深度缓冲的写入,只需要设置它的深度掩码为 GL_FALSE 即可。

1// 设置只读的深度缓冲
2glDepthMask(GL_FALSE);

深度测试函数

OpenGL 允许修改深度测试中使用的比较运算符,允许我们控制 OpenGL 什么时候该通过或丢弃一个片段,什么时候更新深度缓冲。

调用 glDepthFunc 函数设置比较运算符。

1glDepthFunc(GL_LESS);

函数

描述

GL_ALWAYS

永远通过深度测试

GL_NEVER

永远不通过深度测试

GL_LESS

在片段深度值小于缓冲的深度值时通过测试

GL_EQUAL

在片段深度值等于缓冲区的深度值时通过测试

GL_LEQUAL

在片段深度值小于等于缓冲区的深度值时通过测试

GL_GREATER

在片段深度值大于缓冲区的深度值时通过测试

GL_NOTEQUAL

在片段深度值不等于缓冲区的深度值时通过测试

GL_GEQUAL

在片段深度值大于等于缓冲区的深度值时通过测试

默认情况下使用的是 GL_LESS,它将丢弃深度值大于当前深度缓冲值的所有片段。

对于如下代码的组合:

1glEnable(GL_DEPTH_TEST);
2glDepthFunc(GL_ALWAYS);

其效果等效于没有开启深度测试。

但我们开启深度测试之后,就可以得到正常的立方体绘制了。

深度值精度

上面提到的作为比较的深度缓冲,它是位于 0.0 ~ 1.0 之间的深度值,它会与要绘制的物体的 z 值进行比较。

要绘制物体的 z 值就是在运用透视投影或者正交投影视时,介于近平面和远平面之间的任何值。

要把这个 z 值转换为 OpenGL 中的深度值,也就是介于 0.0 和 1.0 之间的值。

有公式如下:

对于这种转换,可以看到,当物体非常接近近平面时,深度值会接近 0.0,当物体非常接近远平面时,深度值会接近 1.0 ,这种转换是一种线性的深度缓冲转换。

而在实践中,几乎不可能是这样的线性转换。

因为当 z 值很小的时候,非常接近近平面,此时我们的观察也会更加精细,而对于较远的物体,接近远平面了,对于它的观察也会比较粗略。

这就和人眼一样,近处的物体当然看得很清了,如果看不清,走近一点就好了,而对于很远的物体,走远了和走近了看得的结果差别不大。

所以在实际将 z 值转换为深度缓冲值,用到的是非线性的转换方程。

它的效果如下:

可以看到在 z 值位于 1.0 和 2.0 之间时,对应的深度值为 0.0 到 0.5 的区间,这就占据了深度值区间范围的 50 %。而 2.0 之后的范围也才占据了 50 %。

这就给了近处的物体一个很大的深度精度。

对于深度值的区间 0.0 到 1.0 ,其实这个区间的前半部分还是和近平面非常近的,不要以为深度值 0.5 就是位于近平面和远平面之间了,其实还非常接近近平面呢。

关于深度测试,就先说到这了,如果有绘制带有深度层次的内容,可别忘了开启深度测试哦。

关于具体的代码实现,可以参考我的 Github 项目:

https://github.com/glumes/AndroidOpenGLTutorial

本文分享自微信公众号 - 纸上浅谈(glumes_blog),作者:glumes

原文出处及转载信息见文内详细说明,如有侵权,请联系 yunjia_community@tencent.com 删除。

原始发表时间:2018-07-20

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

我来说两句

0 条评论
登录 后参与评论

相关文章

  • Android JNI 之 Bitmap 操作

    在 Android 通过 JNI 去调用 Bitmap,通过 CMake 去编 so 动态链接库的话,需要添加 jnigraphics 图像库。

    glumes
  • 【音视频连载-005】基础学习篇-SDL 加载 YUV 文件并显示

    在前面的文章中已经完成了图片的加载和显示,接下来要做的就是加载 YUV 文件并显示。

    glumes
  • LearnOpenGL 源码在 MAC 上的编译与调试

    这两个网站对于学习 OpenGL 帮助非常大,既可以用作入门的教材,也可以作为工具书,后续进行查漏补缺。

    glumes
  • 北航教授李波:说AI会有低潮就是胡扯,这是人类长期的追求

    镁客网
  • 强化学习系列之九:Deep Q Network (DQN)

    我们终于来到了深度强化学习。 ? 1. 强化学习和深度学习结合 机器学习=目标+表示+优化。目标层面的工作关心应该学习到什么样的模型,强...

    AlgorithmDog
  • 深度工作准则1——工作要深入

    yeedomliu
  • 大咖丨张钹院士:人工智能赶超人类的三大法宝

    大数据文摘
  • iOS - Swift CocoaPods导入OC第三方库

    LinXunFeng
  • 用 Python 向你比个心

    之前写了一篇用 Python 画一个小猪佩奇和哆啦 A 梦,然后最近看到有人用 turtle 画了一个心,觉得挺有意思的,于是把代码复制到本地,再加了个播放音乐...

    伪君子
  • 用 Python 向你比个心

    之前写了一篇用 Python 画一个小猪佩奇和哆啦 A 梦,然后最近看到有人用 turtle 画了一个心,觉得挺有意思的,于是把代码复制到本地,再加了个播放音乐...

    伪君子

扫码关注云+社区

领取腾讯云代金券