写在前面

从这节开始,会接触到很多基本概念,原书我也是读了很多遍,一遍一遍去理解其中的意思,以及他们之间的关系。

概念

顶点数组对象:VAO

顶点缓冲对象:VBO

索引缓冲对象:EBO|IBO

OpenGL是一个3D空间,而屏幕和窗口是2D的,所以OpenGL的大部分工作是将3D坐标转换为2D像素

转换的过程就是由图形渲染管线管理的。

图像渲染管线分为两部分:1.把3D坐标转换为2D坐标。2.把2D坐标转变为有颜色的像素

渲染的几个阶段:

顶点数据->顶点着色器->图元装配->几何着色器->光栅化->片段着色器->测试与混合

渲染的每一个阶段都有自己的程序,该程序是使用OpenGL着色器语言进行编写的,这些小程序叫做着色器(shader)

顶点着色器:把3D坐标转换为另一种3d坐标,同时对顶点属性进行一些基本处理

图元装配:将所有的点装配成指定的图元形状

几何着色器:通过产生新的点和新的图元来生成形状

光栅化:将图元映射为最终屏幕上相应的像素,生成片段着色器使用的片段,并裁切吊超过视图以外的像素

片段着色器:计算一个像素的最终颜色

测试与混合:检测片段对应的深度值,用来判断这个像素是其它物体的前面还是后面,决定是否丢弃

代码创建

首先,我们声明出来下面需要使用的着色器源码。

通过上述概念我们知道着色器,是运行在图像渲染管线上的小程序,代码风格类似c语言

 // 顶点着色器源码
const char *vertexShaderSource =
"#version 330 core\n"
"layout (location = 0) in vec3 aPos;\n"
"void main(){\n"
"gl_Position = vec4(aPos.x, aPos.y, aPos.z, 1.0f);\n"
"}\n\0"; //片段着色器源码
const char *fragmentShaderSource =
"#version 330 core\n"
"out vec4 FragColor;\n"
"void main(){\n"
"FragColor = vec4(1.0f, 0.5f, 0.2f, 1.0f);\n"
"}\n\0";

下边我们使用上述两个着色器,构建一个可以渲染的渲染管线

    //创建顶点着色器
int vertexShader = glCreateShader(GL_VERTEX_SHADER);
// 第一个参数着色器对象,第二个参数传递的源码字符串,第三个参数顶点着色器的源码
glShaderSource(vertexShader, , &vertexShaderSource, NULL);
glCompileShader(vertexShader);
// 判断编译是否成功
int sucess;
char infoLog[];
glGetShaderiv(vertexShader, GL_COMPILE_STATUS, &sucess);
if(!sucess){
glGetShaderInfoLog(vertexShader, , NULL, infoLog);
std::cout<< "ERROR::SHADER::VERTEX::COMPILATION_FAILED\n"<< infoLog << std::endl;
}
    // 片段着色器,创建并编译
int fragmentShader = glCreateShader(GL_FRAGMENT_SHADER);
glShaderSource(fragmentShader, , &fragmentShaderSource, NULL);
glCompileShader(fragmentShader);
glGetShaderiv(fragmentShader, GL_COMPILE_STATUS, & sucess);
if(!sucess){
glGetShaderInfoLog(fragmentShader, , NULL, infoLog);
std::cout<<"ERROR::SHADER::FRAGMENT::COMPILATION_FAILED\n"<< infoLog<<std::endl;
}

两个着色器创建,且编译之后,需要进行连接

//链接两个着色器对象到用来渲染的着色器程序中
// 创建着色器程序
int shaderProgram = glCreateProgram();
//链接
glAttachShader(shaderProgram, vertexShader);
glAttachShader(shaderProgram, fragmentShader);
glLinkProgram(shaderProgram);
glGetShaderiv(shaderProgram, GL_COMPILE_STATUS, & sucess);
if(!sucess){
glGetShaderInfoLog(shaderProgram, , NULL, infoLog);
std::cout<<"ERROR::SHADER::PROGRAM::COMPILATION_FAILED\n"<< infoLog<<std::endl;
}

链接两个着色器到我们刚创建的着色器程序中,且进行了编译,这个时候,我们上面的两个着色器,已经在shaderProgram中了。

 //删除无用的顶点着色器以及片段着色器
glDeleteShader(vertexShader);
glDeleteShader(fragmentShader);

我们知道,渲染的几个阶段,那么现在我们只需要创建顶点数据,然后使用上述着色器程序进行渲染

     // 定义三个顶点
float vertices[] = {
-0.5f, -0.5f, 0.0f,
0.5f, -0.5f, 0.0f,
0.0f, 0.5f, 0.0f
};

对于顶点的管理,我们使用顶点缓冲对象以及顶点数组对象进行管理。文章开头已经讲解了基本的概念

// 使用顶点缓冲对象进行缓存
unsigned int VBO, VAO;
glGenVertexArrays(, &VAO);
glGenBuffers(, &VBO);
// 缓冲对象进行绑定
glBindVertexArray(VAO);
glBindBuffer(GL_ARRAY_BUFFER, VBO);

缓冲对象的使用

     // 将定义的顶点数据复制到缓冲内存中
// 第一个参数,缓冲类型,第二个参数数据大小,第三个参数发送的数据,第四个参数,显卡如何管理给定的数据
glBufferData(GL_ARRAY_BUFFER, sizeof(vertices), vertices, GL_STATIC_DRAW); // 解析顶点数据
glVertexAttribPointer(, , GL_FLOAT, GL_FALSE, * sizeof(float), (void*));
// 启动顶点数据
glEnableVertexAttribArray(); glBindBuffer(GL_ARRAY_BUFFER, );
glBindVertexArray(); glDrawArrays(GL_TRIANGLES,,);

现在我们的顶点数组对象已经在内存中了

下面就是调用着色器程序进行启动

         //启用三角形
glClearColor(0.2f, 0.3f, 0.3f, 1.0f);
glClear(GL_COLOR_BUFFER_BIT);
// 激活程序
glUseProgram(shaderProgram);
glBindVertexArray(VAO);
glDrawArrays(GL_TRIANGLES, , );
glfwSwapBuffers(window);

最后,不要忘了清楚我们使用的vao以及vbo对象

//清除对象
glDeleteVertexArrays(, &VAO);
glDeleteBuffers(, &VBO);

总结

通过上述的流程,应该是可以跑起来一个三角形的。OpenGL之路漫漫长,我们继续加油

OpenGL(3)-三角形的更多相关文章

  1. OpenGL 用三角形模拟生成球面

    在看OpenGL红皮书,看到生成球体这节,讲了很多,总感觉不如自己动手写一些代码来的实在,用OpenGL中三角形模拟球形生成.主要要点,模型视图变换,多边形表面环绕一致性,矩阵堆栈.先贴上代码. 虽然 ...

  2. 【OpenGL】三角形

    步骤 初始化顶点数组对象VAO 分配顶点缓冲对象VBO 将顶点数据载入缓冲对象中 glBufferData() 链接顶点属性 glVertexAttribPointer(指定了顶点着色器的变量与我们存 ...

  3. opengl绘制三角形

    顶点数组对象:Vertex Array Object,VAO 顶点缓冲对象:Vertex Buffer Object,VBO 索引缓冲对象:Element Buffer Object,EBO或Inde ...

  4. 1.opengl绘制三角形

    顶点数组对象:Vertex Array Object,VAO,用于存储顶点状态配置信息,每当界面刷新时,则通过VAO进行绘制. 顶点缓冲对象:Vertex Buffer Object,VBO,通过VB ...

  5. CSharpGL(31)[译]OpenGL渲染管道那些事

    CSharpGL(31)[译]OpenGL渲染管道那些事 +BIT祝威+悄悄在此留下版了个权的信息说: 开始 自认为对OpenGL的掌握到了一个小瓶颈,现在回头细细地捋一遍OpenGL渲染管道应当是一 ...

  6. OpenGL2-绘制三角形

    代码下载 /*** 该例子展示如何使用OpenGL绘制三角形* 为什么说绘制三角形呢 ?三维空间里面,我们看到的机会大多数* 漂亮的模型,建筑,任务,机会都是有三角形网络组成.可以说三角形* 是组成三 ...

  7. 03->OpenGL多边形,glut实现三角形条带和三角形扇

    图形学中基本图元是多边形,一般要求是凸多边形,三角形是最简单的凸多边形,在图形渲染中比一般多边形其绘制速度快.今天学习OpenGL绘制三角形条带和三角形扇基础.编程环境! 1. 三角形条带 指定顶点序 ...

  8. 2.通过QOpenGLWidget绘制三角形

    参考:1.opengl绘制三角形 1.QOpenGLWidget的早先版本 QGLWidget是遗留Qt OpenGL模块的一部分,和其他QGL类一样,应该在新的应用程序中避免使用.相反,从Qt 5. ...

  9. [Modern OpenGL系列(三)]用OpenGL绘制一个三角形

    本文已同步发表在CSDN:http://blog.csdn.net/wenxin2011/article/details/51347008 在上一篇文章中已经介绍了OpenGL窗口的创建.本文接着说如 ...

随机推荐

  1. 邮件客户端修改密码—OWA

    邮件客户端修改密码—OWA 1.登录OWA 2.输入用户名 3.点击选项 4.更改密码

  2. PowerShell “execution of scripts is disabled on this system.”

    Set-ExecutionPolicy RemoteSigned

  3. Alpha冲刺报告(6/12)(麻瓜制造者)

    今日已完成 邓弘立: 看github上的开源库 确定了几个对UI改进有帮助的第三方库 符天愉: 部署了用户修改信息,修改头像的接口,并且完成两个接口的api文档,复习了PHP的无限分类来实现商品的发布 ...

  4. css实现常用的两栏三栏布局

    1.两栏 <div class="wrapper"> <div class="half left">left box <p> ...

  5. iosclient发现_世界杯送流量活动项目总结

       世界杯如火如荼的进行.视频站点相似于门户站点.须要高速依据外部环境更新内容. 产品经理须要策划活动,并安排实施.这个活动就是在这样背景下产生的,爱奇艺与运营商合作,实现双赢.爱奇艺能够通过运营商 ...

  6. BZOJ2095:[POI2010]Bridges(最大流,欧拉图)

    Description YYD为了减肥,他来到了瘦海,这是一个巨大的海,海中有n个小岛,小岛之间有m座桥连接,两个小岛之间不会有两座桥,并且从一个小岛可以到另外任意一个小岛.现在YYD想骑单车从小岛1 ...

  7. [JLOI2013]删除物品

    嘟嘟嘟 只要每一次将优先级最高的上面的物品移走,就一定能保证是最优解. 所以我们只要想办法简化这个模拟移物品的过程,看完了题解后,发现可以这么想,我们可以把两个栈头碰头的挨在一起,然后设一个指针代表两 ...

  8. 利用 Settings Sync 同步vs code配置

    vs code上有各种各样不同的插件,如果要在不同的电脑上使用 vs code 配置是件比较麻烦的事情,使用 Settings Sync 将 vs code 配置备份起来,当需要在其他电脑使用  vs ...

  9. C#游戏开发中快速的游戏循环

    C#游戏开发中快速的游戏循环的实现.参考<精通C#游戏编程>一书. using System; using System.Collections.Generic; using System ...

  10. OpenCV 中轮廓包裹的几个函数boundingRect、minAreaRect、minEnclosingCircle用法

    当我们得到对象轮廓后,可用boundingRect()得到包覆此轮廓的最小正矩形,minAreaRect()得到包覆轮廓的最小斜矩形,minEnclosingCircle()得到包覆此轮廓的最小圆形, ...