C++ OpenGL 环境搭建
在 Windows 下使用 Visual Studio 开发 OpenGL,核心在于解决依赖管理。我们主要需要三个部分:IDE、窗口库和函数加载器。
1. 准备工具链
首先安装 Visual Studio,记得勾选 C++ 桌面开发 工作负载,这是编译的基础。
接着是 GLFW,一个轻量级的窗口和输入处理库。下载预编译包后,将 include 和 lib-vc2022 目录直接放入项目根目录即可,这样后续引用头文件会方便很多。
最后是 GLAD,它负责动态加载 OpenGL 的函数指针。访问在线生成器,选择 OpenGL 3.3 Core Profile,生成的 zip 解压后包含 include 和 src 文件夹,这部分代码必须编译进你的项目。
2. 项目配置
新建一个 C++ 空项目,右键属性进行以下设置:
- VC++ 目录:在包含目录中添加
GLFW\include和GLAD\include;库目录添加GLFW\lib-vc2022。 - 链接器 → 输入:附加依赖项填入
glfw3.lib和opengl32.lib。
这一步很关键,配置错了后面编译会报一堆找不到符号的错误。
核心代码实现
环境搭好后,我们来看一段能跑起来的完整示例。这段代码的目标是在屏幕上画一个橙色的三角形,虽然简单,但涵盖了 OpenGL 的核心流程。
#include <glad/glad.h>
#include <GLFW/glfw3.h>
#include <iostream>
// 顶点着色器源码
const char* vertexShaderSource = R"(
#version 330 core
layout (location = 0) in vec3 aPos;
void main() {
gl_Position = vec4(aPos.x, aPos.y, aPos.z, 1.0);
}
);
// 片段着色器源码
const char* fragmentShaderSource = R"(
#version 330 core
out vec4 FragColor;
void main() {
FragColor = vec4(1.0, 0.5, 0.2, 1.0); // 橙色
}
);
void framebuffer_size_callback(GLFWwindow* window, int width, int height)
{
glViewport(0, 0, width, height);
}
int main()
{
// 初始化 GLFW
if (!glfwInit()) {
std::cerr << "GLFW 初始化失败" << std::endl;
return -1;
}
// 设置 OpenGL 版本为 3.3 Core Profile
glfwWindowHint(GLFW_CONTEXT_VERSION_MAJOR, 3);
glfwWindowHint(GLFW_CONTEXT_VERSION_MINOR, 3);
glfwWindowHint(GLFW_OPENGL_PROFILE, GLFW_OPENGL_CORE_PROFILE);
// 创建窗口
GLFWwindow* window = glfwCreateWindow(800, 600, "OpenGL Demo", NULL, NULL);
if (!window) {
std::cerr << "窗口创建失败" << std::endl;
glfwTerminate();
return -1;
}
glfwMakeContextCurrent(window);
glfwSetFramebufferSizeCallback(window, framebuffer_size_callback);
// 加载 OpenGL 函数
if (!gladLoadGLLoader((GLADloadproc)glfwGetProcAddress)) {
std::cerr << "GLAD 初始化失败" << std::endl;
return -1;
}
// 编译着色器
unsigned int vertexShader = glCreateShader(GL_VERTEX_SHADER);
glShaderSource(vertexShader, 1, &vertexShaderSource, NULL);
glCompileShader(vertexShader);
unsigned int fragmentShader = glCreateShader(GL_FRAGMENT_SHADER);
glShaderSource(fragmentShader, 1, &fragmentShaderSource, NULL);
glCompileShader(fragmentShader);
// 创建着色器程序
unsigned int shaderProgram = glCreateProgram();
glAttachShader(shaderProgram, vertexShader);
glAttachShader(shaderProgram, fragmentShader);
glLinkProgram(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 // 顶部
};
// 创建 VAO/VBO
unsigned int VAO, VBO;
glGenVertexArrays(1, &VAO);
glGenBuffers(1, &VBO);
glBindVertexArray(VAO);
glBindBuffer(GL_ARRAY_BUFFER, 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);
glBindVertexArray(0);
// 渲染循环
while (!glfwWindowShouldClose(window)) {
// 输入检测
if (glfwGetKey(window, GLFW_KEY_ESCAPE) == GLFW_PRESS)
glfwSetWindowShouldClose(window, true);
// 渲染
glClearColor(0.2f, 0.3f, 0.3f, 1.0f);
glClear(GL_COLOR_BUFFER_BIT);
glUseProgram(shaderProgram);
glBindVertexArray(VAO);
glDrawArrays(GL_TRIANGLES, 0, 3);
// 交换缓冲
glfwSwapBuffers(window);
glfwPollEvents();
}
// 清理资源
glDeleteVertexArrays(1, &VAO);
glDeleteBuffers(1, &VBO);
glDeleteProgram(shaderProgram);
glfwTerminate();
return 0;
}

