C语言游戏开发:Pygame、SDL、OpenGL深度解析

C语言游戏开发:Pygame、SDL、OpenGL深度解析

C语言游戏开发:Pygame、SDL、OpenGL深度解析

在这里插入图片描述

一、前言:为什么游戏开发是C语言开发的重要技能?

学习目标

  • 理解游戏开发的本质:编写程序实现游戏逻辑、图形渲染、用户交互
  • 明确游戏开发的重要性:支撑游戏产业的发展,成为游戏开发者的必备技能
  • 掌握本章学习重点:Pygame、SDL、OpenGL的开发方法、避坑指南、实战案例分析
  • 学会使用C语言开发游戏,实现游戏逻辑和用户交互

重点提示

💡 游戏开发是C语言开发的重要技能!随着游戏产业的发展,游戏开发的需求越来越大,C语言的高性能和可移植性使其在游戏开发中具有重要地位。


二、模块1:Pygame游戏开发基础

2.1 学习目标

  • 理解Pygame的本质:基于SDL的Python游戏库,简化游戏开发
  • 掌握Pygame的核心架构:窗口管理、事件处理、图形渲染、音频播放
  • 掌握Pygame的开发方法:使用Pygame库进行游戏开发
  • 掌握Pygame的避坑指南:避免窗口创建失败、避免图形渲染错误、避免事件处理错误
  • 避开Pygame使用的3大常见坑

2.2 Pygame的核心架构

窗口管理:创建和管理窗口
事件处理:处理键盘、鼠标、触摸等输入事件
图形渲染:使用Surface进行图形绘制
音频播放:播放音频文件

2.3 Pygame的开发方法

代码示例1:Pygame窗口创建

import pygame import sys # 初始化Pygame pygame.init()# 设置窗口大小和标题 screen_width =800 screen_height =600 screen = pygame.display.set_mode((screen_width, screen_height)) pygame.display.set_caption("Pygame窗口创建")# 设置背景颜色 background_color =(255,255,255)# 游戏循环 running =Truewhile running:# 事件处理for event in pygame.event.get():if event.type== pygame.QUIT: running =False# 填充背景颜色 screen.fill(background_color)# 更新显示 pygame.display.flip()# 退出Pygame pygame.quit() sys.exit()

代码示例2:Pygame绘制图形

import pygame import sys # 初始化Pygame pygame.init()# 设置窗口大小和标题 screen_width =800 screen_height =600 screen = pygame.display.set_mode((screen_width, screen_height)) pygame.display.set_caption("Pygame绘制图形")# 设置颜色 white =(255,255,255) red =(255,0,0) blue =(0,0,255) green =(0,255,0)# 游戏循环 running =Truewhile running:# 事件处理for event in pygame.event.get():if event.type== pygame.QUIT: running =False# 填充背景颜色 screen.fill(white)# 绘制矩形 rect = pygame.Rect(300,200,200,200) pygame.draw.rect(screen, red, rect)# 绘制圆形 pygame.draw.circle(screen, blue,(400,300),100)# 绘制线条 pygame.draw.line(screen, green,(100,100),(700,500),5)# 更新显示 pygame.display.flip()# 退出Pygame pygame.quit() sys.exit()

三、模块2:SDL游戏开发基础

3.1 学习目标

  • 理解SDL的本质:Simple DirectMedia Layer,跨平台图形、音频、输入库
  • 掌握SDL的核心架构:窗口管理、事件处理、图形渲染、音频播放
  • 掌握SDL的开发方法:使用SDL 2.0库进行游戏开发
  • 掌握SDL的避坑指南:避免窗口创建失败、避免图形渲染错误、避免事件处理错误
  • 避开SDL使用的3大常见坑

3.2 SDL的核心架构

窗口管理:创建和管理窗口
事件处理:处理键盘、鼠标、触摸等输入事件
图形渲染:使用渲染器绘制图形
音频播放:播放音频文件

3.3 SDL的开发方法

代码示例3:SDL窗口创建

#include<SDL.h>intmain(int argc,char*argv[]){ SDL_Window *window =NULL; SDL_Renderer *renderer =NULL;// 初始化SDLif(SDL_Init(SDL_INIT_VIDEO)<0){fprintf(stderr,"SDL初始化失败:%s\n",SDL_GetError());return0;}// 创建窗口 window =SDL_CreateWindow("SDL窗口", SDL_WINDOWPOS_CENTERED, SDL_WINDOWPOS_CENTERED,800,600, SDL_WINDOW_SHOWN );if(window ==NULL){fprintf(stderr,"窗口创建失败:%s\n",SDL_GetError());SDL_Quit();return0;}// 创建渲染器 renderer =SDL_CreateRenderer(window,-1, SDL_RENDERER_ACCELERATED);if(renderer ==NULL){fprintf(stderr,"渲染器创建失败:%s\n",SDL_GetError());SDL_DestroyWindow(window);SDL_Quit();return0;}// 设置渲染器颜色SDL_SetRenderDrawColor(renderer,255,255,255,255);SDL_RenderClear(renderer);SDL_RenderPresent(renderer);// 事件循环 SDL_Event event;int quit =0;while(!quit){while(SDL_PollEvent(&event)){if(event.type == SDL_QUIT){ quit =1;}}}// 清理资源SDL_DestroyRenderer(renderer);SDL_DestroyWindow(window);SDL_Quit();return0;}

代码示例4:SDL绘制图形

#include<SDL.h>intmain(int argc,char*argv[]){ SDL_Window *window =NULL; SDL_Renderer *renderer =NULL;// 初始化SDLif(SDL_Init(SDL_INIT_VIDEO)<0){fprintf(stderr,"SDL初始化失败:%s\n",SDL_GetError());return0;}// 创建窗口 window =SDL_CreateWindow("SDL绘制图形", SDL_WINDOWPOS_CENTERED, SDL_WINDOWPOS_CENTERED,800,600, SDL_WINDOW_SHOWN );if(window ==NULL){fprintf(stderr,"窗口创建失败:%s\n",SDL_GetError());SDL_Quit();return0;}// 创建渲染器 renderer =SDL_CreateRenderer(window,-1, SDL_RENDERER_ACCELERATED);if(renderer ==NULL){fprintf(stderr,"渲染器创建失败:%s\n",SDL_GetError());SDL_DestroyWindow(window);SDL_Quit();return0;}// 绘制图形SDL_SetRenderDrawColor(renderer,255,255,255,255);SDL_RenderClear(renderer);// 绘制矩形 SDL_Rect rect ={300,200,200,200};SDL_SetRenderDrawColor(renderer,255,0,0,255);SDL_RenderFillRect(renderer,&rect);// 绘制圆形int x =400, y =300, r =100;SDL_SetRenderDrawColor(renderer,0,0,255,255);for(int i = x - r; i <= x + r; i++){for(int j = y - r; j <= y + r; j++){int dx = i - x;int dy = j - y;if(dx * dx + dy * dy <= r * r){SDL_RenderDrawPoint(renderer, i, j);}}}// 绘制线条SDL_SetRenderDrawColor(renderer,0,255,0,255);SDL_RenderDrawLine(renderer,100,100,700,500);SDL_RenderPresent(renderer);// 事件循环 SDL_Event event;int quit =0;while(!quit){while(SDL_PollEvent(&event)){if(event.type == SDL_QUIT){ quit =1;}}}// 清理资源SDL_DestroyRenderer(renderer);SDL_DestroyWindow(window);SDL_Quit();return0;}

四、模块3:OpenGL游戏开发基础

4.1 学习目标

  • 理解OpenGL的本质:跨平台图形库,用于3D和2D图形渲染
  • 掌握OpenGL的核心架构:上下文管理、着色器、顶点缓冲、纹理
  • 掌握OpenGL的开发方法:使用OpenGL 3.3+核心模式进行游戏开发
  • 掌握OpenGL的避坑指南:避免上下文创建失败、避免着色器编译错误、避免渲染错误
  • 避开OpenGL使用的3大常见坑

4.2 OpenGL的核心架构

上下文管理:创建和管理OpenGL上下文
着色器:顶点着色器和片段着色器,负责图形渲染
顶点缓冲:存储顶点数据
纹理:存储图像数据

4.3 OpenGL的开发方法

代码示例5:OpenGL三角形绘制

#include<stdio.h>#include<stdlib.h>#include<GL/glew.h>#include<GLFW/glfw3.h>const GLchar *vertexSource ="#version 330 core\n""layout (location = 0) in vec3 position;\n""void main() {\n"" gl_Position = vec4(position, 1.0);\n""}\n";const GLchar *fragmentSource ="#version 330 core\n""out vec4 color;\n""void main() {\n"" color = vec4(1.0f, 0.5f, 0.2f, 1.0f);\n""}\n";intmain(){// 初始化GLFWif(!glfwInit()){fprintf(stderr,"GLFW初始化失败!\n");return0;}// 配置GLFWglfwWindowHint(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三角形绘制",NULL,NULL);if(window ==NULL){fprintf(stderr,"窗口创建失败!\n");glfwTerminate();return0;}glfwMakeContextCurrent(window);// 初始化GLEWif(glewInit()!= GLEW_OK){fprintf(stderr,"GLEW初始化失败!\n");glfwTerminate();return0;}// 设置视口int width, height;glfwGetFramebufferSize(window,&width,&height);glViewport(0,0, width, height);// 顶点数据 GLfloat vertices[]={-0.5f,-0.5f,0.0f,0.5f,-0.5f,0.0f,0.0f,0.5f,0.0f};// 创建顶点缓冲对象 GLuint VBO, VAO;glGenVertexArrays(1,&VAO);glGenBuffers(1,&VBO);glBindVertexArray(VAO);glBindBuffer(GL_ARRAY_BUFFER, VBO);glBufferData(GL_ARRAY_BUFFER,sizeof(vertices), vertices, GL_STATIC_DRAW);// 编译顶点着色器 GLuint vertexShader =glCreateShader(GL_VERTEX_SHADER);glShaderSource(vertexShader,1,&vertexSource,NULL);glCompileShader(vertexShader); GLint success; GLchar infoLog[512];glGetShaderiv(vertexShader, GL_COMPILE_STATUS,&success);if(!success){glGetShaderInfoLog(vertexShader,512,NULL, infoLog);fprintf(stderr,"顶点着色器编译失败:%s\n", infoLog);}// 编译片段着色器 GLuint fragmentShader =glCreateShader(GL_FRAGMENT_SHADER);glShaderSource(fragmentShader,1,&fragmentSource,NULL);glCompileShader(fragmentShader);glGetShaderiv(fragmentShader, GL_COMPILE_STATUS,&success);if(!success){glGetShaderInfoLog(fragmentShader,512,NULL, infoLog);fprintf(stderr,"片段着色器编译失败:%s\n", infoLog);}// 链接着色器程序 GLuint shaderProgram =glCreateProgram();glAttachShader(shaderProgram, vertexShader);glAttachShader(shaderProgram, fragmentShader);glLinkProgram(shaderProgram);glGetProgramiv(shaderProgram, GL_LINK_STATUS,&success);if(!success){glGetProgramInfoLog(shaderProgram,512,NULL, infoLog);fprintf(stderr,"着色器程序链接失败:%s\n", infoLog);}glDeleteShader(vertexShader);glDeleteShader(fragmentShader);// 配置顶点属性glVertexAttribPointer(0,3, GL_FLOAT, GL_FALSE,3*sizeof(GLfloat),(GLvoid*)0);glEnableVertexAttribArray(0);glBindBuffer(GL_ARRAY_BUFFER,0);glBindVertexArray(0);// 渲染循环while(!glfwWindowShouldClose(window)){glfwPollEvents();// 清除颜色缓冲glClearColor(0.2f,0.3f,0.3f,1.0f);glClear(GL_COLOR_BUFFER_BIT);// 使用着色器程序glUseProgram(shaderProgram);// 绑定顶点数组对象glBindVertexArray(VAO);// 绘制三角形glDrawArrays(GL_TRIANGLES,0,3);glBindVertexArray(0);// 交换缓冲glfwSwapBuffers(window);}// 清理资源glDeleteVertexArrays(1,&VAO);glDeleteBuffers(1,&VBO);glDeleteProgram(shaderProgram);// 终止GLFWglfwTerminate();return0;}

五、模块4:实战案例分析——使用SDL实现简单的游戏

5.1 学习目标

  • 掌握使用SDL实现简单的游戏:通过SDL库实现一个简单的贪吃蛇游戏
  • 学会使用SDL的窗口管理、渲染、事件处理功能
  • 避开实战案例使用的3大常见坑

5.2 使用SDL实现简单的游戏

代码示例6:贪吃蛇游戏

#include<SDL.h>#include<stdio.h>#include<stdlib.h>#include<time.h>#defineSCREEN_WIDTH800#defineSCREEN_HEIGHT600#defineGRID_SIZE20#defineGRID_WIDTH(SCREEN_WIDTH / GRID_SIZE)#defineGRID_HEIGHT(SCREEN_HEIGHT / GRID_SIZE)#defineINITIAL_SNAKE_LENGTH3#defineSNAKE_SPEED100typedefstruct{int x;int y;} Point;typedefstruct{ Point head; Point body[100];int length;int direction;// 0: up, 1: right, 2: down, 3: left} Snake;typedefstruct{ Point position;int active;} Food;voidinit_snake(Snake *snake){ snake->length = INITIAL_SNAKE_LENGTH; snake->direction =1;// right snake->head.x = GRID_WIDTH /2; snake->head.y = GRID_HEIGHT /2;for(int i =1; i < snake->length; i++){ snake->body[i -1].x = snake->head.x - i; snake->body[i -1].y = snake->head.y;}}voiddraw_snake(SDL_Renderer *renderer, Snake *snake){SDL_SetRenderDrawColor(renderer,0,255,0,255); SDL_Rect head_rect ={snake->head.x * GRID_SIZE, snake->head.y * GRID_SIZE, GRID_SIZE, GRID_SIZE};SDL_RenderFillRect(renderer,&head_rect);for(int i =0; i < snake->length -1; i++){ SDL_Rect body_rect ={snake->body[i].x * GRID_SIZE, snake->body[i].y * GRID_SIZE, GRID_SIZE, GRID_SIZE};SDL_RenderFillRect(renderer,&body_rect);}}voidupdate_snake(Snake *snake,int*game_over){for(int i = snake->length -2; i >=0; i--){ snake->body[i +1]= snake->body[i];} snake->body[0]= snake->head;switch(snake->direction){case0:// up snake->head.y--;break;case1:// right snake->head.x++;break;case2:// down snake->head.y++;break;case3:// left snake->head.x--;break;}if(snake->head.x <0|| snake->head.x >= GRID_WIDTH || snake->head.y <0|| snake->head.y >= GRID_HEIGHT){*game_over =1;}for(int i =0; i < snake->length -1; i++){if(snake->head.x == snake->body[i].x && snake->head.y == snake->body[i].y){*game_over =1;}}}voidgenerate_food(Food *food, Snake *snake){int valid_position =0;while(!valid_position){ food->position.x =rand()% GRID_WIDTH; food->position.y =rand()% GRID_HEIGHT; valid_position =1;if(food->position.x == snake->head.x && food->position.y == snake->head.y){ valid_position =0;}for(int i =0; i < snake->length -1; i++){if(food->position.x == snake->body[i].x && food->position.y == snake->body[i].y){ valid_position =0;}}} food->active =1;}voiddraw_food(SDL_Renderer *renderer, Food *food){if(food->active){SDL_SetRenderDrawColor(renderer,255,0,0,255); SDL_Rect rect ={food->position.x * GRID_SIZE, food->position.y * GRID_SIZE, GRID_SIZE, GRID_SIZE};SDL_RenderFillRect(renderer,&rect);}}voidcheck_collision(Snake *snake, Food *food){if(snake->head.x == food->position.x && snake->head.y == food->position.y){ snake->length++; food->active =0;generate_food(food, snake);}}intmain(int argc,char*argv[]){ SDL_Window *window =NULL; SDL_Renderer *renderer =NULL;if(SDL_Init(SDL_INIT_VIDEO)<0){fprintf(stderr,"SDL初始化失败:%s\n",SDL_GetError());return0;} window =SDL_CreateWindow("贪吃蛇游戏", SDL_WINDOWPOS_CENTERED, SDL_WINDOWPOS_CENTERED, SCREEN_WIDTH, SCREEN_HEIGHT, SDL_WINDOW_SHOWN );if(window ==NULL){fprintf(stderr,"窗口创建失败:%s\n",SDL_GetError());SDL_Quit();return0;} renderer =SDL_CreateRenderer(window,-1, SDL_RENDERER_ACCELERATED);if(renderer ==NULL){fprintf(stderr,"渲染器创建失败:%s\n",SDL_GetError());SDL_DestroyWindow(window);SDL_Quit();return0;}srand(time(NULL)); Snake snake;init_snake(&snake); Food food; food.active =0;generate_food(&food,&snake);int game_over =0;int last_time =SDL_GetTicks();int frame_time =0; SDL_Event event;while(!game_over){int current_time =SDL_GetTicks();int delta_time = current_time - last_time; last_time = current_time; frame_time += delta_time;while(SDL_PollEvent(&event)){if(event.type == SDL_QUIT){ game_over =1;}elseif(event.type == SDL_KEYDOWN){switch(event.key.keysym.sym){case SDLK_UP:if(snake.direction !=2){ snake.direction =0;}break;case SDLK_RIGHT:if(snake.direction !=3){ snake.direction =1;}break;case SDLK_DOWN:if(snake.direction !=0){ snake.direction =2;}break;case SDLK_LEFT:if(snake.direction !=1){ snake.direction =3;}break;}}}if(frame_time >= SNAKE_SPEED){update_snake(&snake,&game_over);check_collision(&snake,&food); frame_time =0;}SDL_SetRenderDrawColor(renderer,0,0,0,255);SDL_RenderClear(renderer);draw_snake(renderer,&snake);draw_food(renderer,&food);SDL_RenderPresent(renderer);SDL_Delay(16);// 限制帧率在60FPS}SDL_DestroyRenderer(renderer);SDL_DestroyWindow(window);SDL_Quit();return0;}

六、本章总结与课后练习

6.1 总结

Pygame游戏开发:基于SDL的Python游戏库,简化游戏开发
SDL游戏开发:跨平台图形、音频、输入库,适合游戏开发
OpenGL游戏开发:跨平台图形库,用于3D和2D图形渲染
实战案例分析:使用SDL实现简单的贪吃蛇游戏

6.2 课后练习

  1. 编写程序:使用Pygame创建窗口并绘制图形
  2. 编写程序:使用Pygame处理键盘和鼠标事件
  3. 编写程序:使用Pygame播放音频
  4. 编写程序:使用SDL创建窗口并绘制图形
  5. 编写程序:使用SDL处理键盘和鼠标事件
  6. 编写程序:使用SDL播放音频
  7. 编写程序:使用OpenGL绘制三角形
  8. 编写程序:使用OpenGL绘制矩形
  9. 编写程序:使用OpenGL加载并显示纹理
  10. 编写程序:使用SDL和OpenGL结合开发3D游戏

Read more

【C++:C++11收尾】解构C++可调用对象:从入门到精通,掌握function包装器与bind适配器包装器详解

【C++:C++11收尾】解构C++可调用对象:从入门到精通,掌握function包装器与bind适配器包装器详解

🎬 个人主页:艾莉丝努力练剑 ❄专栏传送门:《C语言》《数据结构与算法》《C/C++干货分享&学习过程记录》 《Linux操作系统编程详解》《笔试/面试常见算法:从基础到进阶》《Python干货分享》 ⭐️为天地立心,为生民立命,为往圣继绝学,为万世开太平 🎬 艾莉丝的简介: 🎬 艾莉丝的C++专栏简介: 文章目录 * C++学习阶段的三个参考文档 * 8 ~> 包装器 * 8.1 function * 8.1.1 结构 * 8.1.2 概念 * 8.1.3 function实现 * 8.1.4 重写逆波兰表达式求值 * 8.2 bind

By Ne0inhk
【C++】菱形继承为何会引发二义性?虚继承如何破解?

【C++】菱形继承为何会引发二义性?虚继承如何破解?

🎬 个人主页:MSTcheng · ZEEKLOG 🌱 代码仓库 :MSTcheng · Gitee 🔥 精选专栏: 《C语言》 《数据结构》 《C++由浅入深》 💬座右铭:路虽远行则将至,事虽难做则必成! 前言:在之前的文章中,我们已经介绍了继承的相关内容,但有些重要概念尚未涉及,例如菱形继承、虚继承以及二义性等问题。本文将重点探讨这些内容。加粗样式 文章目录 * 一、多继承及菱形继承问题 * 1.1单继承 * 1.2多继承 * 1.3虚继承 * 1.3.1为什么通过虚继承可以将Person部分成员提取出来? * 1.3.2虚继承的构造初始化顺序 * 二、总结 一、多继承及菱形继承问题 1.1单继承 单继承:⼀个派⽣类只有⼀个直接基类时称这个继承关系为单继承。 第一种情形: #include<

By Ne0inhk
【C++】unordered系列容器使用及封装

【C++】unordered系列容器使用及封装

目录 一、unordered_map和unordered_set的使用 1. unordered_set系列的使用 1.1 unordered_set和unordered_multiset参考文档 1.2 unordered_set类的介绍 1.3 unordered_set和set的使用差异 1.4 unordered_map和map的使用差异 1.5 unordered_multimap/unordered_multiset 1.6 unordered_xxx的哈希相关接口 二、用哈希表封装myunordered_map和myunordered_set 1. 源码及框架分析 2. 模拟实现unordered_map和unordered_set 2.1 实现出复用哈希表的框架,并支持insert 2.

By Ne0inhk
扒透 STL 底层!map/set 如何封装红黑树?迭代器逻辑 + 键值限制全手撕----《Hello C++ Wrold!》(23)--(C/C++)

扒透 STL 底层!map/set 如何封装红黑树?迭代器逻辑 + 键值限制全手撕----《Hello C++ Wrold!》(23)--(C/C++)

文章目录 * 前言 * map和set的封装 * 底层红黑树的模拟实现 * 迭代器的模拟实现 前言 你是不是也有过这种 “知其然不知其所以然” 的困惑: 用 map 存键值对、用 set 去重排序时很顺手,但一被问 “map 的 [] 怎么既插入又访问”“set 为啥不能改元素”“它们底层的红黑树到底存的啥”,就瞬间卡壳?甚至看 STL 源码时,被 “KeyOfT”“迭代器 ++ 逻辑” 绕得晕头转向? 其实 map 和 set 的本质,就是对红黑树的 “定制化封装” —— 红黑树是 “通用骨架”,map 和 set 通过 “提取键的规则(KeyOfT)”“迭代器权限控制”“键值修改限制”,分别适配了 “键值对存储”

By Ne0inhk