Android Lesson Six: An Introduction to Texture Filtering
这节课,我们将介绍不同类型的纹理过滤和怎样使用它们, 你将学习如何使纹理看起来更平滑,以及平滑带来的缺点。 |
强烈建议您先阅读Android OpenGL ES 2.0(四)---纹理基础,理解纹理映射在OpenGL中的基本使用。
OpenGLES中的纹理由被称为纹素(texels)的数组组成的,其中包含颜色和alpha值。这与显示器相对应,显示器由一堆像素组成,并在每个点显示不同的颜色。在OpenGL中纹理被用在三角形上并绘制到屏幕,因此这些纹理能绘制出各种各样的尺寸和方向。OpenGL中的纹理过滤选项告诉它如何根据具体情况将纹理像素过滤到设备的像素上。
有三种情况:
OpenGL允许我们为放大和缩小设置过滤器,并允许我们使用最邻近(nearest-neighbour)、双线性(bilinear filtering)和三线性过滤(trilinear filtering)。我们将在下面解释这些意思。
下面是使用最邻近(nearest-neighbour)过滤方式来渲染的放大和缩小的视觉效果,当您将USB连接到Android设备时将会显示可爱的安卓图标:
正常的图片
放大
正如您所见,纹素现在很容易看到,因为当前一个纹素覆盖了很多像素展示出来。
缩小
随着缩小,因为许多纹素不能渲染到可用的有限像素上,所以会导致许多细节都丢失。
双线性插值(Bilinear interpolation)
当纹素值之间没有插值时,在放大示例中,纹理的纹素清晰可见为大正方形。当使用最邻近(nearest-neighbour)方式时,像素将会分配到最邻近的纹素。
通过切换到双线性插值,渲染质量显著提高。这些值将会在邻近的四个像素之间线性插值,而不是将一组像素分配给邻近相同的纹素值。每个像素被平滑化,使得最后的图片看起来也更平滑:
一些块效果仍然很明显,但是这个图片看起来比之前更加平滑。那些在3D加速卡出现前玩过3D游戏的人将会记得软件渲染游戏和硬件加速游戏之间的特性:软件渲染游戏根本没有进行预计算处理,所以一切都显示得块状和锯齿状。一旦人们开始使用图形加速,这些东西都将变得平滑。
双线性插值大多使用在放大。它也能使用在缩小,但是超过某个度,我们将会遇到同样的问题,我们在尝试将太多的纹素放到相同的像素上。OpenGL仅使用最多4个纹素渲染一个像素,因此许多信息仍然会丢失。
如果我们看应用了双线性插值的纹理,当我们在远处看它移动时看起来会很嘈杂,因为每帧都会选择不同的纹素。
纹理映射(Mipmapping)
我们如何才能在缩小纹理时不引用嘈杂并使用上所有纹素呢?我们可以生成一组优化后的不同尺寸的纹理,然后在我们运行的时候使用它们。由于这些纹理已预先生成,它们能使用更多高昂的技术去过滤所有纹素,并且在运行时OpenGL会根据纹理在屏幕上的最终大小选择最合适的层。
生成的图片可以具有更多细节,更少噪点,并且整体上看起来更好。尽管需要更多的内存,但渲染速度也会更快,因为较小的层级能更容易保存在GPU的纹理缓存中。让我们来仔细研究一下原尺寸的1/8倍的图片,在使用了双线性过滤使用纹理映射和双线性过滤没有使用映射。为了清楚图片已被扩大:
没有mipmap的双线性过滤
使用mipmap进行双线性过滤
使用mipmap的版本拥有更多细节,由于图像预处理到单独的层级,所有纹素最终都会在最终的图像中使用。
三线性过滤(Trilinear filtering)
当使用双线性过滤的mipmap时,有时在渲染场景中可以看到明显的跳跃或线,由于OpenGL在纹理的不同mipmap层级之间切换。比较不同的OpenGL纹理的过滤模式将在下面进一步指出。
三线性插值通过在不同mipmap层级之间插值来解决这个问题,这样总共8个纹素将用于插值得到最终的像素值,使得图像更平滑。
OpenGL有两个可以设置的参数:
这些对应于前面描述的缩小和放大。
GL_TEXTURE_MIN_FILTER参数的值可以是下面中的任何一个:
GL_NEAREST
GL_LINEAR
GL_NEAREST_MIPMAP_NEAREST
GL_NEAREST_MIPMAP_LINEAR
GL_LINEAR_MIPMAP_NEAREST
GL_LINEAR_MIPMAP_LINEAR
GL_TEXTURE_MAG_FILTER 参数的值可以是下面中的任何一个:
GL_NEAREST
GL_LINEAR
GL_NEAREST 对应最邻近渲染;
GL_LINEAR 对应双线性过滤;
GL_LINEAR_MIPMAP_NEAREST 对应双线性过滤+mipmap;
GL_LINEAR_MIPMAP_LINEAR 对应三线性过滤;
本课程中将进一步介绍图形示例和最常见选项的进一步说明。
我们首先需要绑定纹理,然后我们可以在该纹理上设置适当的过滤参数:
GLES20.glBindTexture(GLES20.GL_TEXTURE_2D, mTextureHandle);
GLES20.glTexParameteri(GLES20.GL_TEXTURE_2D, GLES20.GL_TEXTURE_MIN_FILTER, filter);
这真的很容易!在加载纹理到OpenGL中后,纹理仍然是绑定的,我们可以简单的调用:
GLES20.glGenerateMipmap(GLES20.GL_TEXTURE_2D);
这将为我们生成所有mipmap级别,并且这些级别将根据纹理过滤器集自动使用。
以下是可用的最常见组合的屏幕截图。 当它在运动时你去看它,效果更加引人注目,所以我建议你下载应用程序并观察它。
最邻近(Nearest-neighbour)渲染
这个模式让人想起旧版3D游戏软件的渲染。
GL_TEXTURE_MIN_FILTER = GL_NEAREST
GL_TEXTURE_MAG_FILTER = GL_NEAREST
双线性过滤,使用mipmap
许多支持3D加速的首批游戏都使用此模式,这是今天在Android手机上平滑纹理的有效方式。
GL_TEXTURE_MIN_FILTER = GL_LINEAR_MIPMAP_NEAREST
GL_TEXTURE_MAG_FILTER = GL_LINEAR
静态图上很难看出图的问题,但是当物体运动时,您可能会注意到渲染的像素在mipmap层级之间切换的水平条带。
三线性过滤
此模式通过在mipmap级别之间进行插值,改进了使用mipmap的双线性滤波的渲染质量。
GL_TEXTURE_MIN_FILTER = GL_LINEAR_MIPMAP_LINEAR
GL_TEXTURE_MAG_FILTER = GL_LINEA
像素在近距离和远距离之间完全平滑;事实上,纹理现在可能在倾斜角度下显示的过于平滑。
各向异性过滤(Anisotropic filtering)是一种更先进的技术,受到某些移动GPU的支持,可用于改善最终结果,超出三线性过滤所能提供的效果。
使用其他模式可以达到什么样的效果?例如,您何时会使用像GL_NEAREST_MIPMAP_LINEAR这样的东西?
可以从GitHub上的项目站点下载本课程的完整源代码。