7、OpenGL初探之OpenGL图像渲染的深度缓冲区理解及隐患解决

         序言:其实对于学习,其实现在写文章是最好的.因为你现在所学到的东西是由浅入深的.写下你对这个知识点的理解,虽然没有深度.在文章中还会体现出很多疑惑.把这些都记录下来.越到后面你会发现你的知识体系搭建起来了,对之前的疑惑就解开了.如果现在不开始记录,慢慢的就会学了后面忘记前面.这样就要查漏补缺了,学习效率不高. 我建议,先追求完成,再追求完美。

好了,开始上今天的干货,OpenGL图像渲染会遇到深度相关的坑点,深度缓冲区具体的含义和深度测试会出现的隐患解决。


一、了解OpenGL深度缓冲区的含义

1、什么是深度?

深度其实就是该像素点在3D世界中距离观察者(比如摄像机)的距离,也是观察者坐标系的z值

2、什么是深度缓冲区?

深度缓冲区其实就是一块内存区域,专门存储每个像素点(绘制在屏幕上的)深度值,深度值(Z)越大,则离观察者(比如摄像机)越远。

3、为什么需要深度缓冲区?

在不使用深度测试的时候,如果我们先绘制一个离我们比较近的物体,再绘制距离比较远的物体,那么因为距离比较远的这个后绘制,就会把距离近的物体覆盖掉。但是有了深度缓冲区之后,会植物体的顺序就不那么重要了,因为只要存在深度缓冲区,OpenGL都会把像素的深度值写到缓冲区中,除非调用glDepthMask(GL_FALSE).来禁⽌止写⼊入。

4、了解深度测试

深度缓冲区(DepthBuffer)和颜色缓冲区(ColorBuffer)是对应的,颜色缓冲区是存储像素的颜色信息,而深度缓冲区存储像素的深度信息。在确定是否绘制一个物体表面的时候,首先要将表面对应的像素深度值与当前深度缓冲区中的值进行比较,如果大于深度缓冲区的值,则丢弃这部分。否则利用这个像素对应的深度值和颜色值分别更新深度缓冲区和颜色缓冲区,这个过程称为深度测试

5、简单了解一下深度值计算

    深度值一般由16位、24位或者32位值表示,通常是24位,位数越高的话,深度的精确度越高,深度值的范围在[0,1]之间,值越小表示月靠近观察者,值越大表示远离观察者。

    深度缓冲主要是通过计算深度值来比较大小,在深度缓冲区中包含深度值,介于0.0~1.0之间,从观察者看到其内容与场景中的所有对象的z值进行了比较,这些视图中的z值可以投影平头截体的近平面和远平面之间的任意值。我们因此需要些方法转换这些视图空间的z值到[0,1]的范围内,下面的线性方程把z值转换为0.0~1.0之间的值。


图1、深度值转换公式

//⽚元着⾊器

out vec4 color;

float LinearizeDepth(float depth)

{

    float near = 0.1;

    float far = 100.0;

    float z = depth * 2.0 - 1.0; // Back to NDC

    return (2.0 * near) / (far + near - z * (far - near));

}

void main() {

    float depth = LinearizeDepth(gl_FragCoord.z);

    color = vec4(vec3(depth), 1.0f);

}


6、深度测试的使用

深度缓冲区一般由窗口管理系统GLFW创建,深度值一般由16位,24位,32位表示,通常情况下是24位,位数越高,深度精度越好。

开启深度测试的方法是

        glEnable(GL_DEPTH_TEST);

关闭深度测试的方法是

        glDisable(GL_DEPTH_TEST);

需要注意的是,使用深度测试之前需要清除深度缓冲和颜色缓冲。清除深度缓冲区默认值为1.0,表示最⼤大的深度值,深度值的范围为(0,1)之间. 值越⼩小表示越靠近观察者,值越⼤大表示 越远离观察者

        glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);

指定深度测试的判断模式

void glDepthFunc(GLEnum mode);


图2、深度测试的判断模式,一般使用GL_LESS


图3、开启深度测试前,z值小的会被z值大的覆盖掉,展示出来的带缺口


图4、开启深度测试后,z值小的(离观察者近的)不会被z值大的覆盖掉,大于深度缓冲区的值被丢弃,展示出来的不带缺口

7、打开和阻断深度缓冲区的写入

void glDepthMask(GLBool value);

value: GL_TRUE 代表开启深度缓冲区的使用 GL_FALSE 代表关闭深度缓冲区的使用


二、使用深度测试的隐患和问题解决

1、ZFighting闪烁的问题

造成ZFighting闪烁问题的原因:是因为开启深度测试后,OpenGL就不会再去绘制模型被遮挡的部分,这样实现显示的更加真实。但是由于深度缓冲区精度的限制对深度相差非常小的情况下,(例如在同一平面上绘制两次深度值一样的图形),openGL就有可能出现不能正确判断两者的深度值得情况,会导致深度测试的结果不可预测,显示出来的现象就是交错闪烁,也就是这两个画面交错出现的情况。

2、ZFighting闪烁问题的问题解决

第一步:启用Polygon Offset(多边形偏移)

解决方法:让深度值之间产生间隔。如果两个图形之间有间隔,是不是就意味着不会产生干涉,可以理解为在执行深度测试前将立方体的深度值做一些细微的调整,于是就能将两个图形的深度值之间有所区分。

启用代码

glEnable(GL_POLYGON_OFFSET_FULL)

参数列表:

GL_POLYGON_OFFSET_POINT 对应点光栅化模式:GL_POINT 

GL_POLYGON_OFFSET_LINE 对应线光栅化模式:GL_LINE

GL_POLYGON_OFFSET_FILL对应像素光栅化模式:GL_FILL


图5、开启隐藏面消除的GL_FILL光栅化


图6、开启了隐藏面消除后的点光栅化GL_POINT


图7、开启了隐藏面剔除的线光栅化GL_LINE

第二步:指定偏移量

1、通过glPolygonOffset来指定,glPolygonOffset需要两个参数,factor,units

2、每个Fragment的深度值都会增加如下所示的偏移量

Offset = ( m * factor ) + ( r * units);

m : 多边形的深度的斜率的最大值,理解一个多边形越是与近裁剪面平行,m 就越接近于0.

r : 能产生于窗口坐标系的深度值中可分辨的差异最小值.r 是由具体OpenGL 平台指定的一个常量。

3、一个大于0的Offset会把模型推到离观察者更远的位置,相应的一个小于0的Offset会把模型拉近。

4、一般而言,只需要将-1.0和-1这样简单的值赋值给glPolygonOffset 基本可以满⾜足需求.

void glPolygonOffset(Glfloat factor,Glfloat units);

应⽤用到⽚段上总偏移计算⽅程式:

Depth Offset = (DZ * factor) + (r * units); DZ:深度值(Z值)

r:使得深度缓冲区产生变化的最小值

负值,将使得z值距离我们更近,⽽正值,将使得z值距离我们更远,我们一般设置factor和units为-1,-1。

第三步、关闭Polygon Offset

glDisable(GL_POLYGON_OFFSET_FILL)

3、ZFighting闪烁问题的预防

讲了这么久,多边形偏移只是一个解决深度测试隐患的方案,当然能从根源上预防ZFighting闪烁问题才是我们开发中需要注意的。下面是预防ZFighting闪烁问题的三个方案

1、不要将两个物体靠的太近,避免渲染时三角形叠在一起。这种方式要求对场景中物体插入一个少量的偏移比如(将其中一个平面向下偏移0.001f),那么就可能避免ZFighting现象。

2、尽可能将近的裁剪面设置得离观察者远一些。离近裁剪平面越近,深度的精确度会越高 ,因此尽可能让近裁剪⾯远一些,那么整个裁剪范围内的精确度就会变高。但是这种⽅式会使离观察者较近的物体被裁减掉,因此需要调试好裁剪面参数。 

3、尽量使用更高位数的深度缓冲区,通常使用的深度缓冲区是24位的,现在有一些硬件使用32位的缓冲 区,使精确度得到提高。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,271评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,275评论 2 380
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,151评论 0 336
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,550评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,553评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,559评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,924评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,580评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,826评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,578评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,661评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,363评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,940评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,926评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,156评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,872评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,391评论 2 342