SDL基础
SDL
SDL(Simple DirectMedia Layer)是一个开源的跨平台多媒体开发库,主要用于开发需要图形、音频和输入设备支持的应用程序。它使用C语言编写,提供了简单易用的API,**能够帮助开发者快速实现跨平台的多媒体功能。**SDL广泛应用于游戏开发、模拟器、媒体播放器等领域。
SDL的主要特点包括:
- 跨平台支持:兼容Windows、macOS、Linux、iOS、Android等多种操作系统。
- 多媒体功能:提供图形渲染、音频处理、输入设备管理等功能。
- 简单易用:API设计简洁,易于上手,适合初学者和专业开发者。
- 社区支持:拥有活跃的开源社区,提供丰富的文档、教程和扩展库。
视频播放器:结合SDL的窗口和渲染功能以及FFmpeg的解码能力,可以开发出跨平台的视频播放器。
实验一 创建窗口
SDL_Init(int flag):初始化SDL系统
好的!以下是将您提供的SDL初始化标志整理成表格的形式,方便查看和理解:
| 标志名称 | 描述 |
|---|---|
SDL_INIT_TIMER | 初始化定时器子系统。 |
SDL_INIT_AUDIO | 初始化音频子系统。 |
SDL_INIT_VIDEO | 初始化视频子系统(包括窗口和渲染器)。 |
SDL_INIT_JOYSTICK | 初始化游戏手柄子系统。 |
SDL_INIT_HAPTIC | 初始化力反馈设备子系统。 |
SDL_INIT_GAMECONTROLLER | 初始化游戏控制器子系统。 |
SDL_INIT_EVENTS | 初始化事件子系统。 |
SDL_INIT_EVERYTHING | 初始化所有子系统。 |
SDL_CreateWindow():创建窗口SDL_Window
SDL_Window* SDL_CreateWindow(const char* title, int x, int y, int w, int h, Uint32 flags)
title:窗口的标题,显示在窗口的标题栏上。
x:窗口在屏幕上的水平位置。可以使用SDL_WINDOWPOS_UNDEFINED让系统自动选择位置。
y:窗口在屏幕上的垂直位置。可以使用SDL_WINDOWPOS_UNDEFINED让系统自动选择位置。
w:窗口的宽度,以像素为单位。
h:窗口的高度,以像素为单位。
当然可以,以下是您需要的表格:
| 标志名称 | 描述 |
|---|---|
SDL_WINDOW_SHOWN | 创建窗口后立即显示。 |
SDL_WINDOW_FULLSCREEN | 创建全屏窗口。 |
SDL_WINDOW_BORDERLESS | 创建无边框窗口。 |
SDL_WINDOW_RESIZABLE | 创建可调整大小的窗口。 |
SDL_WINDOW_MAXIMIZED | 创建最大化窗口。 |
SDL_WINDOW_MINIMIZED | 创建最小化窗口。 |
SDL_WINDOW_HIDDEN | 创建隐藏窗口。 |
**SDL_Delay():**工具函数,用于延时
ms:暂停的时间,以毫秒为单位。
**SDL_Quit():**退出SDL系统
代码
printf("Hello World!\n");SDL_Window *window = NULL; // 声明窗口SDL_Init(SDL_INIT_VIDEO); // 初始化SDL// 创建SDL Windowwindow = SDL_CreateWindow("Basic Window",SDL_WINDOWPOS_UNDEFINED,SDL_WINDOWPOS_UNDEFINED,640,480,SDL_WINDOW_OPENGL | SDL_WINDOW_RESIZABLE);if(!window) // 检测是否创建成功{printf("Can't create window, err:%s\n", SDL_GetError());return 1;}SDL_Delay(10000); // 延迟10000msSDL_DestroyWindow(window); // 消耗窗口SDL_Quit(); // 释放资源return 0;
当设置为 SDL_WINDOWPOS_UNDEFINED 时,表示窗口的初始位置由操作系统决定,通常会根据当前的桌面布局和系统设置自动放置窗口
实验二 渲染器和纹理器
渲染器(Renderer)
渲染器是SDL中用于将图像绘制到窗口或屏幕上的核心组件。它提供了一个统一的接口,允许开发者使用硬件加速来渲染2D图形。渲染器的主要功能包括:
- 创建纹理:渲染器可以创建纹理对象,用于存储图像数据。
- 绘制纹理:渲染器将纹理绘制到窗口上。
- 清除屏幕:渲染器可以清除屏幕内容,为新的绘制操作做准备。
- 更新显示:渲染器将绘制的内容显示到屏幕上。
SDL_Renderer* SDL_CreateRenderer(SDL_Window* window, int index, Uint32 flags);
window:指向 SDL_Window 对象的指针,指定渲染器将在哪个窗口上绘制。
index:指定渲染驱动程序的索引。通常使用 -1,表示使用默认的渲染驱动程序。
flags:指定渲染器的创建标志,用0就行
纹理(Texture)
纹理是存储在显存中的图像数据,利用硬件加速可以高效地进行绘制。纹理的主要功能包括:
- 存储图像数据:纹理可以存储从文件加载的图像数据,或者动态生成的图像数据。
- 高效渲染:纹理可以被快速渲染到屏幕上,适合频繁更新的场景。
SDL_Texture* SDL_CreateTexture(SDL_Renderer *renderer, Uint32 format, int access, int w, int h);
renderer:指向 SDL_Renderer 对象的指针。
format:像素格式,例如 RGB 或 YUV。
access:指定纹理的访问模式,例如 SDL_TEXTUREACCESS_STREAMING 或 SDL_TEXTUREACCESS_TARGET。
(创建可作为渲染目标的纹理:在调用 SDL_CreateTexture 创建纹理时,将 access 参数设置为 SDL_TEXTUREACCESS_TARGET,即可创建一个可以作为渲染目标的纹理。)
w 和 h:纹理的宽度和高度。
SDL_Rect 一个简单的矩形结构
int main(){int run = 1;SDL_Window *window = NULL;SDL_Renderer *renderer = NULL;SDL_Texture *texture = NULL;SDL_Rect rect; // 长方形,原点在左上角rect.w = 50; //方块大小rect.h = 50;SDL_Init(SDL_INIT_VIDEO);//初始化函数,可以确定希望激活的子系统window = SDL_CreateWindow("2 Window",SDL_WINDOWPOS_UNDEFINED,SDL_WINDOWPOS_UNDEFINED,640,480,SDL_WINDOW_OPENGL | SDL_WINDOW_RESIZABLE);// 创建窗口if (!window){return -1;}renderer = SDL_CreateRenderer(window, -1, 0);//基于窗口创建渲染器if (!renderer){return -1;}texture = SDL_CreateTexture(renderer,SDL_PIXELFORMAT_RGBA8888,SDL_TEXTUREACCESS_TARGET,640,480); //创建纹理if (!texture){return -1;}int show_count = 0;while (run){rect.x = rand() % 600;rect.y = rand() % 400;SDL_SetRenderTarget(renderer, texture); // 设置渲染目标为纹理SDL_SetRenderDrawColor(renderer, 0, 0, 0, 255); // 纹理背景为黑色SDL_RenderClear(renderer); //清屏->之前显示清掉SDL_RenderDrawRect(renderer, &rect); //绘制一个长方形SDL_SetRenderDrawColor(renderer, 0, 255, 0, 255); //长方形为白色 rgb ->透明成都 SDL_RenderFillRect(renderer, &rect);SDL_SetRenderTarget(renderer, NULL); //恢复默认,渲染目标为窗口SDL_RenderCopy(renderer, texture, NULL, NULL); //拷贝纹理到CPUSDL_RenderPresent(renderer); //输出到目标窗口上SDL_Delay(300);if(show_count++ > 30){run = 0; // 不跑了}}SDL_DestroyTexture(texture);SDL_DestroyRenderer(renderer);SDL_DestroyWindow(window); //销毁窗口SDL_Quit();return 0;}
一个窗口如果只有一个渲染器,那么整个窗口都是渲染器范围
如果在使用 SDL_RenderCopy(renderer, texture, NULL, NULL); 绘制纹理时省略了源矩形(srcrect)和目标矩形(dstrect),SDL会默认执行以下操作:
源矩形(srcrect):默认为 NULL,表示使用纹理的整个区域。 目标矩形(dstrect):默认为
NULL,表示纹理会被拉伸或压缩以适应渲染目标(窗口)的尺寸。
实验三 事件检测
SDL(Simple DirectMedia Layer)的事件系统是处理用户输入和系统事件的核心机制。它允许开发者检测和响应各种事件,如键盘按键、鼠标移动、窗口调整大小等
操作事件
- SDL_WaitEvent():等待一个事件
- SDL_PushEvent():发送一个事件
事件 - SDL_QUIT:用户点击窗口关闭按钮时触发。
- SDL_KEYDOWN、SDL_KEYUP:键盘按键按下或释放时触发。
- SDL_MOUSEBUTTONDOWN、SDL_MOUSEBUTTONUP:鼠标按键按下或释放时触发。
- SDL_MOUSEMOTION:鼠标移动时触发。
- SDL_USEREVENT:用户自定义事件,开发者可以通过SDL_PushEvent向队列中插入自定义事件。
#define FF_QUIT_EVENT (SDL_USEREVENT + 2)
int main(){SDL_Window *window = NULL; // Declare a pointerSDL_Renderer *renderer = NULL;SDL_Init(SDL_INIT_VIDEO); // Initialize SDL2// Create an application window with the following settings:window = SDL_CreateWindow("An SDL2 window", // window titleSDL_WINDOWPOS_UNDEFINED, // initial x positionSDL_WINDOWPOS_UNDEFINED, // initial y position640, // width, in pixels480, // height, in pixelsSDL_WINDOW_SHOWN | SDL_WINDOW_BORDERLESS// flags - see below);// Check that the window was successfully createdif (window == NULL){// In the case that the window could not be made...printf("Could not create window: %s\n", SDL_GetError());return 1;}/* We must call SDL_CreateRenderer in order for draw calls to affect this window. */renderer = SDL_CreateRenderer(window, -1, 0);/* Select the color for drawing. It is set to red here. */SDL_SetRenderDrawColor(renderer, 255, 0, 0, 255);/* Clear the entire screen to our selected color. */SDL_RenderClear(renderer);/* Up until now everything was drawn behind the scenes.This will show the new, red contents of the window. */SDL_RenderPresent(renderer);SDL_Event event;int b_exit = 0;for (;;){SDL_WaitEvent(&event);//检测事件switch (event.type){case SDL_KEYDOWN: /* 键盘事件 */switch (event.key.keysym.sym){case SDLK_a:printf("key down a\n");break;case SDLK_s:printf("key down s\n");break;case SDLK_d:printf("key down d\n");break;case SDLK_q:printf("key down q and push quit event\n");SDL_Event event_q;event_q.type = FF_QUIT_EVENT;SDL_PushEvent(&event_q);break;default:printf("key down 0x%x\n", event.key.keysym.sym);break;}break;case SDL_MOUSEBUTTONDOWN: /* 鼠标按下事件 */if (event.button.button == SDL_BUTTON_LEFT){printf("mouse down left\n");}else if(event.button.button == SDL_BUTTON_RIGHT){printf("mouse down right\n");}else{printf("mouse down %d\n", event.button.button);}break;case SDL_MOUSEMOTION: /* 鼠标移动事件 */printf("mouse movie (%d,%d)\n", event.button.x, event.button.y);break;case FF_QUIT_EVENT://printf("receive quit event\n");b_exit = 1;break;}if(b_exit)break;}//destory rendererif (renderer)SDL_DestroyRenderer(renderer);// Close and destroy the windowif (window)SDL_DestroyWindow(window);// Clean upSDL_Quit();return 0;}
在这个例子中,FF_QUIT_EVENT 被定义为用户自定义事件类型,其值是 SDL_USEREVENT + 2。这意味着 FF_QUIT_EVENT 是第二个用户自定义事件(从 SDL_USEREVENT 开始计数)。
实验四 多线程
◼ SDL线程创建: SDL_CreateThread:无需阻塞 就能执行
◼ SDL线程等待: SDL_WaitThead
◼ SDL互斥锁: SDL_CreateMutex/SDL_DestroyMutex
◼ SDL锁定互斥: SDL_LockMutex/SDL_UnlockMutex
◼ SDL条件变量(信号量): SDL_CreateCond/SDL_DestoryCond
◼ SDL条件变量(信号量)等待/通知: SDL_CondWait/SDL_CondSinga //自动解锁
SDL_mutex *s_lock = NULL;
SDL_cond *s_cond = NULL;int thread_work(void *arg)
{SDL_LockMutex(s_lock);printf(" <============thread_work sleep\n");sleep(10); // 用来测试获取锁printf(" <============thread_work wait\n");// 释放s_lock资源,并等待signal。之所以释放s_lock是让别的线程能够获取到s_lockSDL_CondWait(s_cond, s_lock); //另一个线程(1)发送signal和(2)释放lock后,这个函数退出printf(" <===========thread_work receive signal, continue to do ~_~!!!\n");printf(" <===========thread_work end\n");SDL_UnlockMutex(s_lock);return 0;
int main(){s_lock = SDL_CreateMutex();s_cond = SDL_CreateCond();SDL_Thread * t = SDL_CreateThread(thread_work,"thread_work",NULL);//开始执行线程if(!t){printf(" %s",SDL_GetError);return -1;}for(int i = 0;i< 2;i++){sleep(2);printf("main execute =====>\n");}printf("main SDL_LockMutex(s_lock) before ====================>\n");SDL_LockMutex(s_lock); // 获取锁,但是子线程还拿着锁 4 秒printf("main ready send signal====================>\n");printf("main SDL_CondSignal(s_cond) before ====================>\n");SDL_CondSignal(s_cond); // 发送信号,唤醒等待的线程printf("main SDL_CondSignal(s_cond) after ====================>\n");sleep(10);SDL_UnlockMutex(s_lock);// 释放锁,让其他线程可以拿到锁printf("main SDL_UnlockMutex(s_lock) after ====================>\n");SDL_WaitThread(t, NULL);SDL_DestroyMutex(s_lock);SDL_DestroyCond(s_cond);return 0;}
main execute =====>
main execute =====>
main SDL_LockMutex(s_lock) before ============>
<thread_work wait
main ready send signal>
main SDL_CondSignal(s_cond) before ====================>
main SDL_CondSignal(s_cond) after ====================>
main SDL_UnlockMutex(s_lock) after ====================>
<===========thread_work receive signal, continue to do _!!!
<===========thread_work end
相关文章:
SDL基础
SDL SDL(Simple DirectMedia Layer)是一个开源的跨平台多媒体开发库,主要用于开发需要图形、音频和输入设备支持的应用程序。它使用C语言编写,提供了简单易用的API,**能够帮助开发者快速实现跨平台的多媒体功能。**SD…...
十三种通信接口芯片——《器件手册--通信接口芯片》
目录 通信接口芯片 简述 基本功能 常见类型 应用场景 详尽阐述 1 RS485/RS422芯片 1. RS485和RS422标准 2. 芯片功能 3. 典型芯片及特点 4. 应用场景 5. 设计注意事项 6. 选型建议 2 RS232芯片 1. RS232标准 2. 芯片功能 3. 典型芯片及特点 4. 应用场景 5. 设计注意事项 6…...
反转一个字符串
用数组栈实现 void Reverse(char *C, int len) {top -1;for(int i 0; i < len; i){push(C[i]);}for(int i 0; i < len; i){C[i] Top();pop();} } 全部函数 #include <stdio.h> #include <stdlib.h> #include <string.h>#define MAX_SIZE 101int …...
从GPT到Gemini 大模型进化史
从GPT到Gemini:大模型进化史 在过去的几年里,人工智能领域经历了翻天覆地的变化,其中最引人注目的莫过于大规模语言模型的发展。从最初的GPT系列到最近的Gemini,这些模型不仅在技术上取得了重大突破,还在实际应用中展…...
【限流算法】计数器、漏桶、令牌桶算法
1 计数器 使用计数器实现限流,可限制在指定时间间隔内请求数小于阈值的情况,但存在临界问题。如图1-17所示,假设每分钟系统限流500个请求,在XX:00:59时刻系统接收到500个请求,在XX:01:00时刻系统又接收到500个请求&am…...
秘密任务 2.0:如何利用 WebSockets + DTOs 设计实时操作
在之前的文章中,我们探讨了为什么 DTO 是提升 API 效率和安全性的秘密武器。现在,我们进入了一个全新的场景——我们将深入探讨如何通过 WebSockets DTOs 实现实时操作! Agent X 正在进行一项高风险的卧底任务。突然,总部更新了…...
RAII 技术详解
1. 核心概念 定义:RAII(Resource Acquisition Is Initialization,资源获取即初始化)是 C 中通过对象生命周期管理资源的核心机制,核心思想是将资源的获取与对象构造绑定、资源释放与对象析构绑定,确…...
Windows快速切换屏幕/桌面
windows自带的切屏 需要winctrl 小键盘左右键 但是! Windows使用还是键盘加鼠标舒服! 教程 安装autohotkey 代码 ~LWin & LButton::{SendInput "^#{Left}" ; 发送 Win Ctrl Left (切换到左侧虚拟桌面) } ; 使用花括号包裹命令&a…...
SpringAI+DeepSeek大模型应用开发——3 SpringAI简介
SpringAI整合了全球(主要是国外)的大多数大模型,而且对于大模型开发的三种技术架构都有比较好的封装和支持,开发起来非常方便; 不同的模型能够接收的输入类型、输出类型不一定相同。SpringAI根据模型的输入和输出类型…...
使用 Function 来编写策略模式:优雅而高效的设计模式实践
引言:为什么选择策略模式? 策略模式(Strategy Pattern)是行为设计模式中的经典之一,它允许我们定义一系列的算法或操作,并使得它们可以互换使用。策略模式的关键思想是将算法的实现与使用它们的上下文分离…...
Java字符串处理
Java字符串处理全解析:String、StringBuilder与StringBuffer 一、String类基础 1. String的本质 不可变对象:Java中的String对象一旦创建就不能修改底层实现:基于private final char value[]字符数组字符串池:JVM维护的特殊存储…...
JS实现RSA加密
目录 目标 环境 实现RSA加解密 计算RSA加密允许的最大字节长度 目标 使用JS实现RSA加密解密。计算RSA加密允许的最大字节长度。 环境 node-rsa 实现RSA加解密 const NodeRSA require(node-rsa);function getKey() {const keyLength512// 创建 RSA 密钥对const key new …...
MySQL GTID集合运算函数总结
MySQL GTID 有一些运算函数可以帮助我们在运维工作中提高运维效率。 1 GTID内置函数 MySQL 包含GTID_SUBSET、GTID_SUBTRACT、WAIT_FOR_EXECUTED_GTID_SET、WAIT_UNTIL_SQL_THREAD_AFTER_GTIDS 4个内置函数,用于GTID集合的基本运算。 1.1 GTID_SUBSET(set1,set2) …...
从“链主”到“全链”:供应链数字化转型的底层逻辑
1. 制造业与供应链数字化转型的必然性 1.1. 核心概念与战略重要性 制造业的数字化转型,是利用新一代数字技术(如工业互联网、人工智能、大数据、云计算、边缘计算等)对制造业的整体价值链进行根本性重塑的过程。这不仅涉及技术的应用&#…...
学习笔记十五——rust柯里化,看不懂 `fn add(x) -> impl Fn(y)` 的同学点进来!
🧠 Rust 柯里化从零讲透:看不懂 fn add(x) -> impl Fn(y) 的同学点进来! 🍔 一、什么是柯里化?先用一个超好懂的生活比喻 假设你在点一个汉堡: 你说:我要点一个鸡腿汉堡! 店员…...
定制化突围:遨游防爆手机的差异化竞争策略
在石油、化工、矿山等危险作业场景中,随着工业智能化与安全生产需求的升级,行业竞争逐渐从单一产品性能的比拼转向场景化解决方案的深度较量。遨游通讯以九重防爆标准为技术底座,融合多模稳控系统与全景前瞻架构,开辟出"千行…...
【Java学习笔记】进制与进制转换
进制与进制转换 一、进制介绍 二进制:0、1,满 2 进 1,以 0b 或 0B 开头。 十进制:0-9,满 10 进 1。 八进制:0-7,满 8 进 1,以数字 0 开头表示。 十六进制:0-9 及 A(10…...
士兵乱斗(贪心)
问题 B: 士兵乱斗 - USCOJ...
【C++面向对象】封装(下):探索C++运算符重载设计精髓
🔥个人主页 🔥 😈所属专栏😈 每文一诗 💪🏼 年年岁岁花相似,岁岁年年人不同 —— 唐/刘希夷《代悲白头翁》 译文:年年岁岁繁花依旧,岁岁年年看花之人却不相同 目录 C运…...
JVM初探——走进类加载机制|三大特性 | 打破双亲委派SPI机制详解
目录 JVM是什么? 类加载机制 Class装载到JVM的过程 装载(load)——查找和导入class文件 链接(link)——验证、准备、解析 验证(verify)——保证加载类的正确性 准备(Prepare&…...
[图论]Kruskal
Kruskal 本质:贪心,对边进行操作。存储结构:边集数组。适用对象:可为负权图,可求最大生成树。核心思想:最短的边一定在最小生成树(MST)上,对最短的边进行贪心。算法流程:对全体边集…...
UML-饮料自助销售系统(无法找零)序列图
一、题目: 在饮料自动销售系统中,顾客选择想要的饮料。系统提示需要投入的金额,顾客从机器的前端钱币口投入钱币,钱币到达钱币记录仪,记录仪更新自己的选择。正常时记录仪通知分配器分发饮料到机器前端,但可…...
Nginx Http配置整理
一、nginx 配置参数: server {#SSL 默认访问端口号为 443listen 443 ssl;#请填写绑定证书的域名server_name cloud.tencent.com; #请填写证书文件的相对路径或绝对路径ssl_certificate cloud.tencent.com_bundle.crt; #请填写私钥文件的相对路径或绝对路径ssl_cer…...
爬虫利器SpiderTools谷歌插件教程v1.0.0!!!web端JavaScript环境检测!!!
SpiderTools谷歌插件教程v1.0.0 一、SpiderTools简介二、下载通道三、插件介绍四、插件使用五、工具函数使用 一、SpiderTools简介 SpiderTools主要用于检测和监控网页的JavaScript运行环境。该插件可以帮助开发者更好地查看网页运行环境,特别是在处理复杂的前端环…...
计算机视觉算法实战——基于YOLOv8的农田智能虫情测报灯害虫种类识别系统开发指南
✨个人主页欢迎您的访问 ✨期待您的三连 ✨ ✨个人主页欢迎您的访问 ✨期待您的三连 ✨ ✨个人主页欢迎您的访问 ✨期待您的三连✨ 一、智能虫情监测领域概述 1.1 农业虫害防治现状 全球每年因虫害造成的粮食损失达20%-40%,我…...
14-算法打卡-哈希表-基本概念-第十四天
1 基本概念 1.1 哈希表 百度百科解释: 散列表(Hash table,也叫哈希表),是根据关键码值(Key value)而直接进行访问的数据结构。也就是说,它通过把关键码值映射到表中一个位置来访问记录,以加快…...
趣味编程之分布式系统:负载均衡的“雨露均沾“艺术
#此篇文章由Deepseek大力支持😋 凌晨三点,西二旗某火锅店后厨—— “羊肉卷走3号桌!” “肥牛卷去7号!” “虾滑优先给VIP区!” 我蹲在传菜口的监控屏幕前,看着机器人服务生们忙而不乱地穿梭。突然间&am…...
第十六届蓝桥杯大赛软件赛省赛 C++ 大学 B 组 部分题解
赛时参加的是Python组,这是赛后写的题解,还有两题暂时还不会,待更新 题目链接题目列表 - 洛谷 | 计算机科学教育新生态 A 移动距离 答案:1576 C 可分解的正整数 Python3 import itertools from functools import cmp_to_ke…...
考研数据结构之顺序查找、折半查找与分块查找详解(包含真题及解析)
考研数据结构之顺序查找、折半查找与分块查找详解 一、顺序查找(Sequential Search) 1.1 基本思想 顺序查找是最基础的查找算法,通过遍历数据集合逐个比较目标值与当前元素,直到找到匹配项或遍历结束。其核心特点是:…...
英文查重的时候参考文献显示重复是怎么回事?
像上图这样参考文献部分有颜色的情况,是属于参考文献没有排除干净的问题。 如何解决这样的问题? 首先第一步,先确认该报告是不是排除参考文献的版本; 第二步,如果是排除参考文献的版本,且参考文献仍然有…...
