【转】DIB位图(Bitmap)的读取和保存

转自:https://www.cnblogs.com/wangguchangqing/p/5417444.html

设备无关位图(Device Independent Bitmap)是可以保存在磁盘的位图文件,可以从磁盘读取到内存或者从内存保存到磁盘上。它的文件结构是标准化的,可以在Windows/Linux/Unix等平台上显示相同的效果。本文主要介绍了

  1. 如果将位图文件从磁盘读到内存中
  2. 在内存中对位图文件进行操作后,如何将位图保存到磁盘

1 读取位图到内存中

1.1 DIB文件结构

要将位图文件(.bmp)从磁盘读取到内存,首先要了解其文件结构。DIB的文件组成有以下4个部分:

  1. 文件表头,主要包含了文件的类型(必须是BM),文件的大小(所占用的字节数)和位图的像素矩阵的便宜量。
  2. 信息表头,包含了两部分内容:位图的相关信息(位图的大小、位深度、位面数、压缩和编码等)和指向RGB颜色表(调色盘)的指针。
  3. RGB色彩对照表,也就是调色板,不一定会有。16位及以上直接使用RGB通道表示颜色,一般不需要调色板。
  4. 位图的像素信息矩阵,表示具体的像素。1,4,8位颜色,保存的是调色板的索引,具体的颜色根据索引在调色板中查找;16位及其以上不使用调色板,直接使用RGB组成像素颜色。

1.2 在Windows下DIB的内存结构

要将DIB数据读取到内存,就需要在内存中分配相应的空间。Windows提供了几种结构体,结构体中的字段对应着DIB文件的各个信息值,具体如下
alter

引用自 http://blog.csdn.net/wenzhou1219/article/details/26162869

将DIB读取到内存只需要将磁盘数据填充到相应到结构体即可。在磁盘上DIB需要连续的结构存储,在内存中则不需要连续的存储空间,可以分段将数据读取到相应的结构体中。
读取DIB到内存的具体步骤:

  1. 将文件头信息读取到BITMAPFILEHEADER结构体中。
  2. 将位图头信息读取到BITMAPINFOHEADER结构体中。
  3. 如果有调色板,则将其信息读取到RGBQUAD中。
  4. 读取位图像素信息到像素矩阵中。
fp.Read(&bmfileHeader, sizeof(BITMAPFILEHEADER)); // 读取BMP文件头 
...
//读取文件信息头
ret = fp.Read(&bmHeader, sizeof(BITMAPINFOHEADER)); 
...
fp.Read(m_dibBits, GetBodySize());  //读取像素信息 

1.3 结构体各字段信息

BITMAPFILEHEADER

代表文件头信息的结构体BITMAPFILEHEADER的声明如下:

typedef struct tagBITMAPFILEHEADER {WORD    bfType;DWORD   bfSize;WORD    bfReserved1;WORD    bfReserved2;DWORD   bfOffBits;
} BITMAPFILEHEADER  

其中,

  • bfType是文件类型,该字段必须是BM,如果不是则说明该文件不是DIB。
  • bfSize是位图文件的大小(字节数)
  • bfReserved1和bfReserved2 是保留字段
  • bfOffBits 从BITMAPFILEHEADER的起始位置到位图像素的字节偏移量。

BITMAPINFO

在上面提到,位图信息和位图的调色板是存放在同一个结构体中的,该结构体就是BITMAPINFO,其声明如下

typedef struct tagBITMAPINFO {BITMAPINFOHEADER bmiHeader;RGBQUAD          bmiColors[1];
} BITMAPINFO, *PBITMAPINFO;

bmiHeader是位图头信息
bmiColors是调色板

BITMAPINFOHEADER

位图的头信息结构BITMAPINFOHEADER,该结构包含了DIB的尺寸和颜色格式等信息,声明如下

typedef struct tagBITMAPINFOHEADER{DWORD      biSize;LONG       biWidth;LONG       biHeight;WORD       biPlanes;WORD       biBitCount;DWORD      biCompression;DWORD      biSizeImage;LONG       biXPelsPerMeter;LONG       biYPelsPerMeter;DWORD      biClrUsed;DWORD      biClrImportant;
} BITMAPINFOHEADER  

其中,

  • biSize是该结构体所占用的字节说
  • biWidth DIB的宽(以像素为单位),如果biCompression是BI_JPEG或者BI_PNG,则biWidth是解压缩后JPEG或者PNG图像的宽度。
  • biHeight,DIB的高(以像素为单位)。
    • 如果biHeight是正的,则DIB的像素是按照从下往上(bottom-up,也就是像素数组的第一行保存的实际是DIB的最后一行像素值),图像源点在左下角。
    • 如果biHeight是负的,则DIB的像素是按照从上往下保存的(up-bottom),而且像素的数据是不能被压缩的其biCompression必须是BI_RGB或者BI_FIELDS
    • 如果biCompression是BI_JPEG或者BI_PNG,则biHeight是解压缩后JPEG或者PNG的高
  • biPlanes 目标设备的平面数,总是设为1.
  • biBitCount,每个像素所占用的位数。0,表示JPEG或者PNG指定每个像素所占用的位数。还可以是1,4,8,16,24,32。
  • biCompression,数据的压缩方法(up-down的DIB不能被压缩),可以是以下值:
    • BI_RGB / BI_FIELDS未被压缩
    • BI_RLE4 / BI_RLE8 使用游程长度编码 (RLE,run-length encode)
    • BI_JPEG / BI_PNG 指示该图像是JPEG或者PNG图像。
  • biSizeImage 图像的字节数,对于BI_RGB的DIB其值为0.
  • biXPelsPerMeter / biYPelsPerMeter 显示该DIB的目标设备所需的分辨率(单位是像素每米)
  • biClrUsed DIB实际使用调色板中的颜色个数,通常为0表示使用调色板中的全部颜色。
  • biClrImportant 显示DIB所必须的颜色个数,通常为0表示全部颜色都是必须的。

详细的解释参见 https://msdn.microsoft.com/en-us/library/dd183376(v=vs.85).aspx

RGBQUAD

typedef struct tagRGBQUAD {BYTE rgbBlue;BYTE rgbGreen;BYTE rgbRed;BYTE rgbReserved;
} RGBQUAD;

注意,其存储顺序是BGR。

1.3 DIB的结构实例


是一幅宽和高都是32的位图(放大后,可以看到表示像素的一个个方块),其有16种颜色。下面是该图像的16进制数据

  • 首先是文件的头信息 BITMAPFILEINFO 共有14个字节(0x00-0x0D),其起始的两个字节为0x4d42(大端存储,高位在前)表示文件类型为BM,最后4个字节 0x0076是位图的像素信息相对于文件头的偏移量,也就是从0x0076为图像的像素信息。
  • 下面是位图的头信息 BITMAPINFOHEADER共有40个字节(0x0E-0x35),起始的4个字节是0x0028就是该结构体的大小,紧接着的4个字节是图像的宽0x0020
  • 跟着是图像的调色板,0x36 - 0x75。调色板共有16种颜色,也就是说有调色板中有16个RGBQUAD,其大小为16 * 4.
  • 最后是位图的数据 0x76-0x275

1.4 读取

    CFile fp(dibName, CFile::modeRead | CFile::typeBinary);BITMAPFILEHEADER bmfileHeader;BITMAPINFOHEADER bmHeader;ULONGLONG headpos;int paletteSize = 0;int ret, cbHeaderSize;headpos = fp.GetPosition(); // 获取文件指针的位置ret = fp.Read(&bmfileHeader, sizeof(BITMAPFILEHEADER)); // 读取BMP文件头if (bmfileHeader.bfType != 0x4d42) //判断文件类型标头是不是x4d42,表示该文件为BMP类型文件{AfxMessageBox(_T("文件不是bmp!"));return;}//读取文件信息头ret = fp.Read(&bmHeader, sizeof(BITMAPINFOHEADER));// 计算RGBQUAD的大小switch (bmHeader.biBitCount){case 1:paletteSize = 2;break;case 4:paletteSize = 16;break;case 8:paletteSize = 256;break;}// 为BITMAPINFO分配存储空间cbHeaderSize = sizeof(BITMAPINFOHEADER)+paletteSize * sizeof(RGBQUAD);m_dibInfo = (BITMAPINFO*) new char[cbHeaderSize];m_dibInfo->bmiHeader = bmHeader;if (paletteSize) //是否有调色板{ret = fp.Read(&(m_dibInfo->bmiColors[0]), paletteSize * sizeof(RGBQUAD));if (ret != int(paletteSize * sizeof(RGBQUAD) ) ){delete[] m_dibInfo;m_dibInfo = NULL;return;}}//为像素数组分配存储空间,大小由GetBodySize决定m_dibBits = (void*) new char[GetBodySize()];fp.Seek(headpos + bmfileHeader.bfOffBits, CFile::begin); // 将文件指针移动到DIB像素数组ret = fp.Read(m_dibBits, GetBodySize());if (ret != int(GetBodySize())){delete[] m_dibInfo;delete[] m_dibBits;m_dibInfo = NULL;m_dibBits = NULL;}fp.Close();  

知道了DIB的文件结构后,读取其到内存还是挺简单的,需要注意的是DIB的像素数组的大小。由于DIB的宽度需要时4的倍数,不是的话需要填充0将其凑成4的倍数,所以其像素数组的大小不能简单的width * height * biBitCount / 8,其中biBitCount是每个像素占用的位数。其具体的计算方法如下

  1. 首先计算一行所占用的字节数 bytesPerLine
    bytesPerLine = ((m_dibInfo->bmiHeader.biWidth * m_dibInfo->bmiHeader.biBitCount + 31) / 32 ) * 4
  2. 将bytesPerLine乘以图像的高
    bytesPerLine * m_dibInfo->bmiHeader.biHeight

1.5 总结

本文主要对DIB的文件结构以及其对应的内存中的结构体做了一个总结,并对一个具体的DIB16进制数据结构进行分析,最后实现了如何将一个DIB数据读取到内存中。

一直对位图结构不是很了解,趁着在公司实习没有具体的工作安排,对DIB的结构作了个总结。至于如何将处理后的位图数据写回磁盘文件,在知道位图结构的情况下,只需要填充相应的结构字段就行了,需要注意的还是位图的宽需要是4的倍数,不是的话要用0补齐。本来想写个DEMO的但是太累,明天再说吧。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/436043.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

16进制字符串转化为10进制数

同学在MSN CDC电话面试(可惜我在被面试的时候全然没有这么具体的问题了:))中的一个题目:将16进制的字符串转化为10进制数字,例如“1A”,则对应26。题目很简单,实现起来也容易&#x…

arch linux 入门,arch linux 从来就不是给新手入门用的

该楼层疑似违规已被系统折叠 隐藏此楼查看此楼在学生的时候,常会有些科目议题难,大部分的人死命地猛读,却也考得低分;也是会有够聪明又考的高分的人,轻描淡写地说:没时间读,随便看一下&#xff…

【转】DCMTK各模块说明!!!!!!!

转自:https://blog.csdn.net/Kelvin_Yan/article/details/50765693 有删改 原文来自wiki DCMTK:http://support.dcmtk.org/redmine/projects/dcmtk/wiki/modules 各模块说明 These are the modules of the public DCMTK toolkit (version 3.6.0): 关…

IT农民工如何来美国工作

经历一年多的等待与折腾,终于来美国工作了。自打进入IT这一行那天起,就梦想着有一天能在硅谷的顶尖的IT公司上班,没想到梦想就这么成真了。当然,我的梦想不算远大。但是我觉得人的梦想是随着成长一点一点变大的。伟人们的远大理想…

【转】DICOM医学图像处理:基于DCMTK工具包学习和分析worklist

转自:https://blog.csdn.net/zssureqh/article/details/38775315 背景: DICOM3.0协议中有介绍关于worklist的部分。简而言之,worklist可以看做是放射科设备从医院RIS系统中自动读取患者信息的一种“通信协议”,可以指存储在RIS系…

libc.so.6linux查找,Linux中提示:/lib64/libc.so.6: version `GLIBC_2.17' not found 的解决办法...

昨天在服务器上安装好node之后,提示这个错误:./node: /lib64/libc.so.6: version GLIBC_2.17 not found (required by ./node),今天把解决过程整理一下一、查看系统中可使用的glibc版本//使用strings命令查看strings /lib64/libc.so.6 |grep …

linux将汇编转为机器码,汇编语言 高级语言 机器语言 本地代码

不管是什么语言,最终都会转化为机器语言(本地代码)(机器码),计算机程序的运行最终仍是以机器语言(本地代码)(机器码)运行的。java汇编语言:linux汇编语言是低级编程语言,不像高级语言有跨平台性,首先,CPU的…

【转】关于DCMTK中像素存储以及getoutdata()函数的使用

转自:https://blog.csdn.net/ancewer/article/details/73277895 有删改 当你看这个帖子的时候,假设你也因为这个问题而感到困惑。 在使用这个函数的时候纠结了很久,各种google、百度都没查到相关资料,测试了好多次,并…

【Visual C++】游戏开发笔记三十五 站在巨人的肩膀上:游戏引擎导论

看到在留言中很多朋友提到不太清楚DirectX与游戏引擎的区别的问题,在这里浅墨就专门把自己对游戏引擎的一些理解写成一篇文章,作为我们《Visual C游戏开发》专栏的游戏引擎导论,也希望能通过这篇文章,能让大家有所启发&#xff0c…

String ... String 三个点 jdk1.5的特性.才知道

String ... String 三个点 jdk1.5的特性.才知道 String... excludeProperty表示不定参数,也就是调用这个方法的时候这里可以传入多个String对象。public static void main(String[] args) {//测试,传入多个参数test("hello", "world"…

c语言大数相加oj,郑州轻工业大学oj题解(c语言)论如何正确的提高正确率:水题合集(四)...

好像离上一篇水题篇过去已经快半个月了~也是好久没有发过这些简单又有趣的题目了呢,今天想起来就总结几道吧。今天是看龙族的一天~ 花了一下午时间把《龙族2》看了2/3,看书的时间真的过的飞快,听着歌翻着书,一会4.5个小时就过去了…

【转】DICOM:DICOM Print服务中PresentationContext协商之 MetaSOPClass与SOPClass对比分析!!!!!!!!

转自:https://zssure.blog.csdn.net/article/details/45119841 背景: 最近项目中遇到的实际问题较多,且大多是较隐蔽的、不易被发现的错误。究其根源来看,还是对DICOM3.0协议中的细节掌握不够仔细,因而导致在实际编码…

2008秋季-计算机软件基础-未交实验报告名单

实验四 序号:108 实验五 序号:108,225,226 351 实验二 353 实验一 356 实验二 360 实验三 365 实验三 367 实验三 368 实验二 372 实验五 转载于:https://www.cnblogs.com/emanlee/archive/2008/11/02/1324910.html

操作系统真实的虚拟内存是什么样的

1. 内存及虚存基本布局 提起虚存,大都能说出几条来。 对于32位系统,大多数操作系统都会将4GB的内存空间的一部分挪给内核使用,应用程序无法直接访问这一段内存,这部分内存空间称为内核空间。Windows默认情况下会将高地址的2GB分配…

约瑟夫问题c语言链表解法,约瑟夫环问题 --链表 C语言

总共有m个人在圆桌上,依次报名,数到第n个数的人退出圆桌,下一个由退出人下一个开始继续报名,循环直到最后一个停止将编号输出#include #include typedef struct Head * PHead;typedef struct Node * PNode;struct Head{PNode Next…

【转】用fo-dicom实现print scu的注意事项!!!!!!!!!

转自:https://blog.csdn.net/tvsofa2008/article/details/50245357 用fo-dicom实现print scu的注意事项 fo-dicom是一个开源的协议库,开发语言是c#。网上针对fo-dicom的分析也有不少,但是专门针对dicom print的文章还是太少了。 近几天需要…

DELPHI串口通讯编程

spcomm更加简单 转贴 Delphi中串口通信的实现 河南省计算中心 张海航 -------------------------------------------------------------------------------- Delphi是一种具有功能强大、简便易用和代码执行速度快等优点的可视化快速应用开发工具,它在…

设计模式C++实现 ——观察者模式

观察者模式:定义对象间的一种一对多的依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都得到通知并被自动更新。它还有两个别名,依赖(Dependents),发布-订阅(Publish-Subsrcibe)。可以举个博客订阅的例子&…

【转】DCMTK开源库的学习笔记1:将DCM文件保存成BMP文件或数据流(即数组)

转自:https://blog.csdn.net/zssureqh/article/details/8784980 DCMTK开源库介绍: DCMTK是目前最全面实现DICOM3.0标准的开源库,通过结合DCMTK开源库和CxImage图像开源库,能够很方便的开发属于自己的DCM文件编辑浏览软件。 DCMT…

如何将C语言翻译成汇编语言,如何把汇编语言转换成C语言

该楼层疑似违规已被系统折叠 隐藏此楼查看此楼程序:ORG 0000HLJMP MAINORG 000BHLJMP TIMEORG 1000HHOUR1 EQU 10hHOUR2 EQU 12hMIN1 EQU 14hMIN2 EQU 16hSEC1 EQU 18HCOUNT EQU 20HNUM1 EQU 22HNUM2 EQU 24HLL4 EQU 26Hs1 bit P1.0s2 bit P1.1s3 bit P1.2s4 bit P1.3A1 EQU 36…