首页 游戏 软件 资讯 排行榜 专题
首页
编程语言
C++实战教程将内存Bitmap数据保存为BMP文件

C++实战教程将内存Bitmap数据保存为BMP文件

热心网友
93
转载
2026-05-06

C++如何将内存中的Bitmap数据保存为BMP文件【实战】

BMP文件需手动构造BITMAPFILEHEADER和BITMAPINFOHEADER头结构,像素数据按BGR顺序、从下到上存储且每行4字节对齐;24位真彩色推荐biBitCount=24、biCompression=BI_RGB,并须翻转内存行序以匹配BMP左下原点规范。

c++如何将内存中的Bitmap数据保存为BMP文件【实战】

免费影视、动漫、音乐、游戏、小说资源长期稳定更新! 👉 点此立即查看 👈

BITMAPINFOWriteFile 手动写 BMP 文件头与像素数据

想在Windows平台下把内存里的像素数据存成BMP文件?最直接、也最可控的方法,其实不是去调用那些封装好的GDI/GDI+函数,而是亲自动手,按照BMP的二进制格式从头到尾“组装”一遍。BMP文件结构非常清晰:一个文件头(BITMAPFILEHEADER),紧跟着一个信息头(BITMAPINFOHEADER),如果需要的话还有调色板(RGBQUAD,只有1位、4位或8位图才需要),最后就是像素数据本身了。数据顺序是BGR,并且每行必须用零填充到4字节的整数倍。只要你手头已经有了原始的像素数组(比如一个uint8_t*指针),剩下的就是按部就班地拼接数据,然后写入磁盘文件。

这里有几个关键参数必须弄对:BITMAPINFOHEADER里的biWidthbiHeight决定了图像的尺寸;biBitCount通常设为24(表示24位真彩色,省去处理Alpha通道的麻烦)或32;biCompression务必设为BI_RGB,表示未压缩。还有一个极易出错的点:BMP格式规定像素数据从图像的最底下一行开始存储。如果你的内存数据是常规的“从上到下”顺序(比如大多数CPU渲染的结果),那么在写入前,必须手动把行序翻转过来。

  • 计算对齐后的行宽:每行实际占用的字节数 = ((width * biBitCount + 31) / 32) * 4(这个公式确保结果向上对齐到4字节)。
  • 计算文件总大小:文件总字节数 = sizeof(BITMAPFILEHEADER) + sizeof(BITMAPINFOHEADER) + pixel_data_size
  • 写入操作:建议使用Windows API CreateFileWriteFile来执行文件写入。相比C标准库的fwrite,它们能提供更底层的控制,避免潜在的换行符转换或缓冲问题。

24 位 BGR 数据保存时为何图像上下颠倒?

保存出来的图片头朝下了?别慌,这很可能不是你代码的逻辑错误,而是BMP格式本身的“特性”。BMP规范白纸黑字写着:像素数据的第0行,对应的是图片的最底部。如果你的内存数组pixels是按我们习以为常的“第0行是顶部”的光栅顺序排列的,那么直接逐行写入,结果自然就是上下颠倒的。

解决办法很直观:在逐行复制数据时,从源数据的最后一行开始往前读。具体可以这样操作:

立即学习“C++免费学习笔记(深入)”;

for (int y = 0; y < height; ++y) {
    const uint8_t* src_row = pixels + (height - 1 - y) * stride_in_bytes;
    memcpy(dst_row_ptr, src_row, width * 3);
    dst_row_ptr += row_size_padded;
}

这里,stride_in_bytes是你原始内存中每行数据占用的字节数(它可能没有4字节对齐),而row_size_padded则是我们之前计算好的、BMP要求对齐后的行宽。忽略这个翻转步骤,是新手在保存BMP时踩到的最常见的一个“坑”。

BITMAPINFObiSizeImage 字段到底要不要填?

关于biSizeImage这个字段,微软的官方文档说,当压缩方式为BI_RGB且颜色位数不是16位时,你可以把它设为0,系统会自动计算像素数据区的大小。话虽如此,但经验表明,显式地计算并填写这个值是更稳妥的做法。这能避免某些解析器因为字段为0而产生误判或兼容性问题。

需要特别注意的是:biSizeImage仅仅表示像素数据部分的大小,它不包括文件头和信息头。一个常见的混淆点是把它当成了整个文件的大小,或者与文件头中的bfSize(文件总大小)字段搞混了。

  • 正确的关联是:bfSize = sizeof(BITMAPFILEHEADER) + sizeof(BITMAPINFOHEADER) + biSizeImage
  • 如果你填写了biSizeImage,却忘了同步更新bfSize,那么用系统自带的“画图”软件打开时,很可能会报错“不是有效的BMP文件”。
  • 另外,如果你处理的是32位数据(带Alpha通道),需要确保Alpha通道不被写入BMP。标准的BI_RGB格式并不支持Alpha,强行写入会导致兼容性问题。除非使用BI_BITFIELDS,但那又是另一番复杂故事了。

跨平台保存时为什么不用 Gdiplus::Bitmap::Sa ve

你可能会问,既然Windows提供了Gdiplus::Bitmap::Sa ve这样现成的函数,为什么还要大费周章地手动构造呢?原因在于依赖和可控性。使用GDI+意味着你的程序必须依赖gdiplus.dll,并且需要先调用GdiplusStartup进行初始化,这在某些场景下显得笨重且线程不安全。更关键的是,它对内存数据的解释可能不符合你的预期——它默认按BGRA顺序处理,而你传入的可能是纯BGR或RGB数组,一不小心就会导致颜色错乱。此外,它内部隐藏了行对齐的细节,在某些图像尺寸下,可能会悄无声息地添加黑色填充边,而你却无从知晓。

相比之下,纯Win32的手动写法没有任何额外依赖,初始化开销为零,并且让你对每一个字节的布局都有完全的掌控权。唯一的“代价”是多写大约50行C++代码。但话说回来,这50行代码一旦写对,其稳定性和可移植性极高,可以说放之二十年而皆准。

真正的挑战,其实不在于填充那几个结构体,而在于行对齐的计算和上下翻转的逻辑,是否与你原始数据的布局严丝合缝。举个例子,哪怕是一个常见的1920像素宽度,计算(1920*3+31)/32*4得到的是5760。如果这里算错成了5764,就意味着每行会多写4个字节,导致后续所有行的偏移全部错位,整个图像也就面目全非了。细节,才是成败的关键。

来源:https://www.php.cn/faq/2325812.html
免责声明: 游乐网为非赢利性网站,所展示的游戏/软件/文章内容均来自于互联网或第三方用户上传分享,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系youleyoucom@outlook.com。

相关攻略

c++如何解析MPEG-TS流中的PAT与PMT节目表【深度】
编程语言
c++如何解析MPEG-TS流中的PAT与PMT节目表【深度】

C++如何解析MPEG-TS流中的PAT与PMT节目表【深度】 PAT表是解析MPEG-TS流的关键起点,它固定位于PID为0x0000的TS包中。解析时需通过payload_unit_start_indicator标志定位新表起始,正确处理adaptation field以找到payload,校验

热心网友
05.06
C++ std::identity用法 _ 函数对象占位符与ranges算法【详解】
编程语言
C++ std::identity用法 _ 函数对象占位符与ranges算法【详解】

C++ std::identity用法详解:函数对象占位符与ranges算法核心指南 std::identity 核心概念与应用场景解析 在C++20标准库中,std::identity绝非简单的语法糖,而是std::ranges算法体系中表达“元素原样透传”意图的唯一标准函数对象。当你调用std:

热心网友
05.06
C++ std::is_base_of用法 _ 编译期检查类继承关系【干货】
编程语言
C++ std::is_base_of用法 _ 编译期检查类继承关系【干货】

std::is_base_of编译期报错解析:非法类型、不完整类型与非类类型传入的应对方案 std::is_base_of 编译期报错的根本原因 许多C++开发者在首次使用 std::is_base_of 模板时,常对其在编译阶段直接报错感到困惑。这源于其作为类型特征(type trait)的本质—

热心网友
05.06
c++如何读取和设置文件的扩展时间戳信息_出生时间提取【技巧】
编程语言
c++如何读取和设置文件的扩展时间戳信息_出生时间提取【技巧】

Linux下birth time仅能通过statx()读取且不可设置,需内核≥4 11、支持的文件系统及正确挂载选项;glibc未暴露该字段,stat()等传统接口无法获取。 Linux 下用 stat 和 utimensat 读取 设置 birth time(创建时间) 在Linux的世界里,文件

热心网友
05.06
c++ cista++序列化 c++如何进行极低延迟的对象序列化
编程语言
c++ cista++序列化 c++如何进行极低延迟的对象序列化

cista 实现微秒级序列化的核心原理:零开销内存拷贝与偏移重定位 cista 微秒级序列化的技术实现解析 cista 之所以能够实现微秒甚至纳秒级的序列化性能,源于其颠覆性的设计理念。与传统的序列化方案不同,cista 彻底摒弃了运行时类型识别(RTTI)、动态反射和堆内存分配等重型操作。它采用了

热心网友
05.06

最新APP

宝宝过生日
宝宝过生日
应用辅助 04-07
台球世界
台球世界
体育竞技 04-07
解绳子
解绳子
休闲益智 04-07
骑兵冲突
骑兵冲突
棋牌策略 04-07
三国真龙传
三国真龙传
角色扮演 04-07

热门推荐

POE交换机连接设备后频繁重启原因解析
电脑教程
POE交换机连接设备后频繁重启原因解析

Poe交换机带载后重启:是故障,还是系统在“自救”? 不少朋友遇到过这个头疼的问题:PoE交换机一接上设备就重启。其实,这本质上不是设备坏了,而是供电系统一套精密的自我保护机制在起作用。当负载接入的瞬间,如果系统检测到功耗超标、供电不稳等情况,就会主动触发复位,防止硬件受损。这正是IEEE 802

热心网友
05.06
电饼铛选购指南哪款型号性价比最高
电脑教程
电饼铛选购指南哪款型号性价比最高

高性价比电饼铛:精准匹配、扎实可靠、真正省心 挑选一款高性价比的电饼铛,核心其实很明确:功能要精准匹配你的真实需求,材质工艺必须扎实可靠,细节设计能让你每天用着都省心。它追求的绝不是单纯的便宜或者参数漂亮,而是每一分钱都花在刀刃上。比如,2100W级的稳定火力保证了煎烤效率不打折;0氟不粘涂层配合蜂

热心网友
05.06
红米K30 5G动态壁纸不联网可以使用吗
电脑教程
红米K30 5G动态壁纸不联网可以使用吗

红米K30 5G动态壁纸联网机制全解析 关于红米K30 5G的动态壁纸是否需要一直联网,答案是:完全没必要。这玩意儿用起来其实很“懂事”,它只在你第一次上手和偶尔想换新的时候,才需要网络搭把手。 其背后的逻辑很清晰:手机搭载的MIUI系统,把所有酷炫的动态壁纸资源都放在了小米官方的“云端仓库”里。所

热心网友
05.06
vivo Y35手机桌面时间不显示修复方法
电脑教程
vivo Y35手机桌面时间不显示修复方法

vivo Y35桌面时间不显示?别急,这事儿有解 不少vivo Y35用户可能都遇到过这个情况:一觉醒来,或者换个主题之后,主屏幕上那个熟悉的“时间”不见了。先别急着怀疑手机坏了,事实是,超过八成的类似问题,根源其实很简单——时间组件压根没被“请”上桌面,或者相关的自动设置被无意中关闭了。作为一台搭

热心网友
05.06
英雄联盟手游杰斯新皮肤获取方法与实战评测
游戏攻略
英雄联盟手游杰斯新皮肤获取方法与实战评测

英雄联盟手游杰斯新皮肤外观设计酷炫,充满科技感。技能特效以蓝色能量为主,视觉效果震撼且辨识度高。实战中技能清晰、手感流畅,能提升操作自信与战场表现。整体而言,该皮肤在视觉、特效与实战体验上均表现优异,值得玩家入手。

热心网友
05.06