当前位置: 首页 > news >正文

【OpenGL学习】(二)OpenGL渲染简单图形

文章目录

  • 【OpenGL学习】(二)OpenGL渲染简单图形
    • OpenGL渲染图形流程
    • 顶点,图元和片元
    • VAO,VBO ,EBO
    • 着色器
    • 示例:使用OpenGL渲染三角形

【OpenGL学习】(二)OpenGL渲染简单图形

OpenGL渲染图形流程

CPU(中央处理器)↓
初始化窗口与OpenGL上下文↓
编译并链接着色器程序 ↓
准备顶点数据(顶点位置、颜色、法线、纹理坐标等)↓
创建并绑定:- VAO(顶点数组对象)← 用于记录 VBO/EBO 的绑定状态和顶点属性配置↓创建并绑定:- VBO(顶点缓冲对象)← 存储所有顶点属性数据(位置信息、颜色、纹理坐标等)- EBO(索引缓冲对象,可选)← 存储索引数据,减少冗余顶点设置顶点属性指针(glVertexAttribPointer)← 告诉 OpenGL 如何解析 VBO 中的顶点数据启用顶点属性数组(glEnableVertexAttribArray)↑(这些设置会被 VAO 记录下来)调用:- glBufferData → 将顶点数据或索引数据从 CPU 传输到 GPU 显存中(显卡缓冲区)↓
====================  GPU 开始接管渲染流程 ====================
顶点着色器(Vertex Shader)- 每个顶点执行一次- 坐标变换:模型矩阵 × 视图矩阵 × 投影矩阵- 传出数据供后续阶段使用(如颜色、纹理坐标)↓
图元装配(Primitive Assembly)- 将一组顶点组装为图元(如三角形、线段)↓
(可选)几何着色器(Geometry Shader)- 每个图元执行一次- 可动态生成新的顶点或图元↓
光栅化(Rasterization)- 将图元转换为片元(像素候选)- 生成每个片元的屏幕位置↓
片段着色器(Fragment Shader)- 每个片元执行一次- 计算颜色值(光照、纹理采样、颜色混合等)↓
测试与混合阶段(由固定功能单元执行)- 深度测试、模板测试- α混合、遮挡判断↓
帧缓冲(Framebuffer)- 最终图像写入帧缓冲 → 显示在屏幕

在这里插入图片描述
图片来源:https://geekdaxue.co/read/Learn-OpenGL-CN/01-Getting-Started-04-Hello-Triangle.md

顶点,图元和片元

顶点(Vertex)是图形的基本构建单位,表示图形的一个顶点,通常包括:

  • 位置坐标(Position):如三维空间中的 (x, y, z)
  • 颜色信息(Color)
  • 纹理坐标(UV)
  • 法向量(Normal)
  • 其他自定义属性(如切线、位移等)

图元(Primitive)是由多个顶点组成的几何形状单元,比如:

  • 点(GL_POINTS)→ 每个顶点单独成图元
  • 线段(GL_LINES)→ 每两个顶点组成一条线
  • 三角形(GL_TRIANGLES)→ 每三个顶点组成一个三角形
  • 更多复合图元(如 GL_TRIANGLE_STRIP, GL_TRIANGLE_FAN)

片元(Fragment)是每个图元经过光栅化后生成的像素候选者,这些片元会传入片段着色器进行着色计算(颜色、纹理、光照等)。在经过深度测试、模板测试、混合等处理之后,部分片元会变成屏幕上的像素。

VAO,VBO ,EBO

VAO(顶点数组对象,Vertex Array Object)用于 存储顶点属性配置状态,如顶点属性指针、VBO绑定状态、EBO绑定状态等。每次绘制时只需绑定一次 VAO,而不用重复设置顶点属性。

VBO(顶点缓冲对象,Vertex Buffer Object)用于在 GPU 显存中存储顶点数据(如位置、颜色、法线、纹理坐标等),避免每次绘制时从 CPU 向 GPU 频繁传输数据,提升效率。

EBO(元素缓冲对象,Element Buffer Object)用于在 绘制图形时按照索引重用顶点数据。节省存储空间,避免重复顶点。

着色器

着色器(Shader)是运行在 GPU 上的小程序,用于控制图形渲染的每个阶段。它是实现可编程渲染管线的核心。

着色器的主要类型:

着色器类型作用
顶点着色器 (Vertex Shader)处理每个顶点的位置变换、法线、纹理坐标等
片段着色器 (Fragment Shader)处理每个像素(片元)的颜色计算、纹理映射、光照等
几何着色器(可选) (Geometry Shader)处理图元(点、线、三角形),可生成新图元
曲面细分控制/评估着色器(可选)用于对几何细分
计算着色器(Compute Shader)用于并行计算任务,不用于图形绘制

着色器使用的语言是GLSL(OpenGL Shading Language)。GLSL的语法类似于 C 语言,且可以运行于 GPU 上。我们需要在 C++ 中编写 GLSL 代码(字符串形式),然后使用 OpenGL API 编译和链接着色器。

示例:使用OpenGL渲染三角形

#include <glad/glad.h>  // 加载 OpenGL 函数指针
#include <GLFW/glfw3.h> // GLFW 用于创建窗口和处理输入#include <iostream>     // C++ 标准输入输出库// 回调函数声明:当窗口大小发生改变时调用
void framebuffer_size_callback(GLFWwindow* window, int width, int height);
// 处理输入的函数声明
void processInput(GLFWwindow* window);// 设置窗口宽高
const unsigned int SCR_WIDTH = 800;
const unsigned int SCR_HEIGHT = 600;// 顶点着色器GLSL源码
const char* vertexShaderSource = "#version 330 core\n" // 使用 OpenGL 3.3 对应的 GLSL 版本(即 GLSL 3.30)
"layout (location = 0) in vec3 aPos;\n"  // 顶点位置属性,位置值为0
"void main()\n"
"{\n"
"   gl_Position = vec4(aPos.x, aPos.y, aPos.z, 1.0);\n"  // 设置顶点位置
"}\0";// 片段着色器GLSL源码
const char* fragmentShaderSource = "#version 330 core\n"
"out vec4 FragColor;\n"  // 片段输出颜色
"void main()\n"
"{\n"
"   FragColor = vec4(1.0f, 0.5f, 0.2f, 1.0f);\n"  // 设置输出颜色为橙色
"}\n\0";int main()
{// 初始化并配置 GLFWglfwInit();glfwWindowHint(GLFW_CONTEXT_VERSION_MAJOR, 3); // OpenGL 主版本号glfwWindowHint(GLFW_CONTEXT_VERSION_MINOR, 3); // OpenGL 次版本号glfwWindowHint(GLFW_OPENGL_PROFILE, GLFW_OPENGL_CORE_PROFILE); // 使用核心模式#ifdef __APPLE__glfwWindowHint(GLFW_OPENGL_FORWARD_COMPAT, GL_TRUE); // MacOS 需要加这句
#endif// 创建 GLFW 窗口GLFWwindow* window = glfwCreateWindow(SCR_WIDTH, SCR_HEIGHT, "LearnOpenGL", NULL, NULL);if (window == NULL){std::cout << "Failed to create GLFW window" << std::endl;glfwTerminate(); // 初始化失败,退出程序return -1;}glfwMakeContextCurrent(window); // 将窗口上下文设为当前线程上下文glfwSetFramebufferSizeCallback(window, framebuffer_size_callback); // 设置窗口大小改变时的回调// 初始化 GLAD,用于加载 OpenGL 函数if (!gladLoadGLLoader((GLADloadproc)glfwGetProcAddress)){std::cout << "Failed to initialize GLAD" << std::endl;return -1;}// ------------------构建并编译着色器程序------------------// 顶点着色器unsigned int vertexShader = glCreateShader(GL_VERTEX_SHADER); // 创建着色器对象glShaderSource(vertexShader, 1, &vertexShaderSource, NULL);   // 附加源码glCompileShader(vertexShader);                                // 编译着色器// 检查编译错误int success;char infoLog[512];glGetShaderiv(vertexShader, GL_COMPILE_STATUS, &success);if (!success){glGetShaderInfoLog(vertexShader, 512, NULL, infoLog);std::cout << "ERROR::SHADER::VERTEX::COMPILATION_FAILED\n" << infoLog << std::endl;}// 片段着色器unsigned int fragmentShader = glCreateShader(GL_FRAGMENT_SHADER);// 创建一个片段着色器对象glShaderSource(fragmentShader, 1, &fragmentShaderSource, NULL); // 将片段着色器源码附加到着色器对象上glCompileShader(fragmentShader);   // 编译片段着色器源码// 检查编译错误glGetShaderiv(fragmentShader, GL_COMPILE_STATUS, &success);if (!success){glGetShaderInfoLog(fragmentShader, 512, NULL, infoLog);std::cout << "ERROR::SHADER::FRAGMENT::COMPILATION_FAILED\n" << infoLog << std::endl;}// 着色器程序unsigned int shaderProgram = glCreateProgram();           // 创建程序对象glAttachShader(shaderProgram, vertexShader);              // 附加着色器glAttachShader(shaderProgram, fragmentShader);glLinkProgram(shaderProgram);                             // 链接程序// 检查链接错误glGetProgramiv(shaderProgram, GL_LINK_STATUS, &success);if (!success) {glGetProgramInfoLog(shaderProgram, 512, NULL, infoLog);std::cout << "ERROR::SHADER::PROGRAM::LINKING_FAILED\n" << infoLog << std::endl;}// 删除已编译的着色器对象,已链接到程序中不再需要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(1, &VAO); // 创建顶点数组对象glGenBuffers(1, &VBO);      // 创建顶点缓冲对象glBindVertexArray(VAO);     // 绑定 VAOglBindBuffer(GL_ARRAY_BUFFER, VBO); // 绑定 VBO glBufferData(GL_ARRAY_BUFFER, sizeof(vertices), vertices, GL_STATIC_DRAW);// 传入数据// 设置顶点属性指针glVertexAttribPointer(0, 3, GL_FLOAT, GL_FALSE, 3 * sizeof(float), (void*)0);glEnableVertexAttribArray(0); // 启用顶点属性glBindBuffer(GL_ARRAY_BUFFER, 0); // 解绑 VBO,为了安全,非必须glBindVertexArray(0);            // 解绑 VAO// 可以取消注释以使用线框模式绘制:也就是不填充图形,只画出边框线// glPolygonMode(GL_FRONT_AND_BACK, GL_LINE);// 渲染循环while (!glfwWindowShouldClose(window)){// 处理输入processInput(window);// 清屏并设置背景颜色glClearColor(0.2f, 0.3f, 0.3f, 1.0f);glClear(GL_COLOR_BUFFER_BIT);// 绘制三角形glUseProgram(shaderProgram); // 使用着色器程序glBindVertexArray(VAO);      // 绑定 VAOglDrawArrays(GL_TRIANGLES, 0, 3); // 从第0个顶点绘制3个顶点构成的三角形// 交换缓冲区并查询IO事件glfwSwapBuffers(window);glfwPollEvents();}// 可选:释放所有资源glDeleteVertexArrays(1, &VAO);glDeleteBuffers(1, &VBO);glDeleteProgram(shaderProgram);// 释放 GLFW 资源glfwTerminate();return 0;
}// 处理输入:如果按下 ESC 键,则关闭窗口
void processInput(GLFWwindow* window)
{if (glfwGetKey(window, GLFW_KEY_ESCAPE) == GLFW_PRESS)glfwSetWindowShouldClose(window, true);
}// 当窗口大小发生改变时,自动调整视口大小
void framebuffer_size_callback(GLFWwindow* window, int width, int height)
{glViewport(0, 0, width, height); // 设置 OpenGL 视口大小
}

在这里插入图片描述
只画边框线:

参考:
https://learnopengl-cn.github.io/01%20Getting%20started/04%20Hello%20Triangle/

相关文章:

  • 系统架构设计(八):三层架构
  • SVN 版本控制入门指南
  • Qt与Hid设备通信
  • Python多进程编程执行任务
  • Class类的详细说明
  • Go语言 GORM框架 使用指南
  • Unity 人物模型学习笔记
  • Windows 上安装下载并配置 Apache Maven
  • 英语学习5.17
  • 系统架构设计师案例分析题——软件架构设计篇
  • 深入解析 React 的 useEffect:从入门到实战
  • 网络切片:给用户体验做“私人定制”的秘密武器
  • Spring Boot- 2 (数万字入门教程 ):数据交互篇
  • shell脚本之条件判断,循环控制,exit详解
  • NestJS——日志、NestJS-logger、pino、winston、全局异常过滤器
  • JDBC 的编写步骤及原理详解
  • 多指标组合策略
  • 什么情况下使用ActiveMQ
  • 【读代码】端到端多模态语言模型Ultravox深度解析
  • Flask项目实践:构建功能完善的博客系统(含评论与标签功能)
  • 外企聊营商|上海仲裁:化解跨国企业纠纷的“上海路径”
  • 张国清将赴俄罗斯举行中俄“长江—伏尔加河”地方合作理事会第五次会议和“东北—远东”政府间合作委员会双方主席会晤
  • ESG考证虚火:相比证书,知识结构+实战经验更重要
  • 一图看懂|印巴交火后,双方基地受损多少?
  • 中央军委决定调整组建3所军队院校
  • 国台办:民进党当局刻意刁难大陆配偶,这是不折不扣的政治迫害