OpenGL入门

写在前面:

最近项目要做视频编辑功能,需要用到GPUImage,但是GPUImage又要用到OpenGL ES的知识,于是就从最上层一直追代码追到了OpenGL里,发现OpenGL能做的事情太多了!而且代码看起来很复杂,完全的晦涩难懂。被OpenGL里面的语句搞得一个头两个大,一会儿gen一下,一会儿bind一下,一会儿又active一下。这里总结下网上的教程,希望对OpenGL有更好的理解。

先来解释下OpenGL为什么会射击这么多的操作顺序。这是因为和我现在使用的C++,Object-C这种面向对象的语言不同,OpenGL中的大多数函数使用了一种基于状态的方法,大多数的OpenGL对象都需要在使用前把该对象绑定到context上。这里有两个新名词——OpenGL对象Context

Context

Context是个非常抽象的概念,我理解的是有点像iOS中的画布(CGContextRef),也可以把它理解成一个包含了所有OpenGL状态的对象。如果我们把一个Context销毁了,那么OpenGL也不复存在。

OpenGL对象

我们可以吧OpenGL对象理解成一个状态的集合,它负责管理它下属的所有状态。当然,除了状态,OpenGL对象还会存储其他数据。注意,这些状态和context中的状态并不重合,只有把一个OpenGL对象绑定到context上时,OpenGL对象的各种状态才会映射到context的状态。因此,这时如果我们改变了context的状态,那么也会影响这个对象,而相反的,依赖这些context状态的函数也会使用存储这个对象上的数据。

因此,OpenGL对象的绑定既可能是为了修改该对象的状态(大多数对象需要绑定到context上才可以改变它的状态),也可能是为了让context渲染时使用它的状态。

画了一个图,仅供理解。图中灰色的方块代表各种状态,箭头表示把一个OpenGL对象绑定到context上后,对应状态的映射。

OpenGL入门_第1张图片
image.png

OpenGL对象 包含了下面一些类型:Buffer Objects, Vertex Array Objects, Texttures, Framebuffer Objects等等。
我们下面会讲到Vertex Array Objects这个对象。

这些对象都有三个相关的重要函数:

void glGen*(GLsizei n, GLuint *objects);

负责生成一个对象的name。而name就是这个对象的引用。

  void glDelete*(GLsizei n, const GLuint *objects);

负责销毁一个对象。

  void glBind*(GLenum target, GLuint object);

将对象绑定到context上。

关于OpenGL对象还有很多内容,可以参见官方Wiki

这里还要了解一些图形名词。

渲染(Rendering): 计算机从模型到创建一张图像的过程。OpenGL仅仅是其中一个渲染系统。热爱是基于一个光栅化的系统,其他的系统还有光线追踪(但是有时也会用到OpenGL)等。

模型(Models)或者对象(Objects):这里两者含义一样。指从几何图元一一点,线,三角形中创建的东西,由顶点指定。

着色器(Shaders):这是一类特殊的函数,在图形硬件上执行的。我们可以理解成,Shader是一些为图形处理单元(GPU)编译的小程序。OpenGL包含了编译工具来把我们编写的Shader源代码编译成可以在GPU上运行的代码。在OpenGL中,我们可以使用四种Shader阶段。最常见的就是vertext shaders——他们可以处理顶点数据;以及fragment shaders,它们处理光栅化后生成的fragments。vertext shadersfragment shaders是每个OpenGL程序必不可少的部分。

像素(pixel):像素是我们显示器上最小可见元素。我们系统中的像素被存储在一个帧缓存(framebuffer)中。帧缓存是一块由图形硬件管理的内存空间,用于提供给我们的显示设备。

OpenGL坐标系

OpenGL坐标系是不同于UIKit坐标系,它是这样的

OpenGL入门_第2张图片
image.png

除了方向,还有一个点需要注意,默认情况各个方向坐标值范围为(-1, 1),而不是UIKit中的(0, 320)。当绘制点(320, 0),它并不会出现在屏幕右上角。在SE1中,可以通过以下代码将坐标系转化为熟悉的(320, 480)

- (void)glkView:(GLKView *)view drawInRect:(CGRect)rect {
    glViewport(0, 0, rect.size.width * 2, rect.size.height * 2);
    glMatrixMode(GL_PROJECTION);
    glLoadIdentity();
    glOrthof(0, 320, 0, 480, -1024, 1024);
    glMatrixMode(GL_MODELVIEW);
    glLoadIdentity();
}

你可能感兴趣的:(OpenGL入门)