看到几篇关于BMP结构的文章,就转过来:
文章1:
位图文件结构
+---------------------------------------------------------+
| 位图文件头 |
| (BITMAPFILEHEADER:14字节) |
+----+----------------------------------------------------+
| | 位图信息头 |
| 位 | (目前有4个版本:OS/2, 版本3,版本4,版本5) |
| +----------------------------------------------------+
| 图 | 位屏蔽 |
| +----------------------------------------------------+
| 信 | 颜色表 |
| | (只有当每个像素的位数小于等于8时,才存在) |
| 息 | (如果采用颜色表,像素阵列中的像素是颜色表达的索引) |
| | (有三种格式:OS/2(RGBTRIPLE),其它(RGBQUAD))... |
+----+----------------------------------------------------+
| |
| 像素阵列(Pixel[][]) |
| (通常情况下,图像是对其到32位边界的扫描线序列) |
| |
+---------------------------------------------------------+
*********************************************************************************/
/*********************************************************************************
压缩方法
BI_RGB 未压缩
BI_RLE8 使用行程编码压缩的8-bpp图像,只适用于由下向上的DIB
BI_RLE4 使用行程编码压缩的4-bpp图像,只适用于由下向上的DIB
BI_BITFIELDS 16-、32-bpp的未压缩图像,有三个屏蔽用于确定如何从像素中提取RGB成分
BI_JPEG 像素阵列是JPEG格式的嵌入图像,只用于Window98/2000
BI_JPEG 像素阵列是PNG格式的嵌入图像,只用于Window98/2000
*********************************************************************************/
/* 位图文件头 */
typedef struct tagBITMAPFILEHEADER {
WORD bfType; /* 位图文件签名:必须为:0x4D42, 即"BM" */
DWORD bfSize; /* 此位图文件长度 */
WORD bfReserved1; /* 保留 */
WORD bfReserved2; /* 保留 */
DWORD bfOffBits; /* 像素阵列相对文件开头的偏移量 */
} BITMAPFILEHEADER, FAR *LPBITMAPFILEHEADER, *PBITMAPFILEHEADER;
/* 位图信息头:版本3 */
typedef struct tagBITMAPINFOHEADER{
DWORD biSize; /* 信息头长度:识别各种版本的唯一方法 */
LONG biWidth; /* 像素宽度 */
LONG biHeight; /* 像素高度,正:由下向上,负:由上至下 */
WORD biPlanes; /* 位平面,必须为1 */
WORD biBitCount; /* 每个像素的位数,如256色,8bit */
DWORD biCompression; /* 压缩方法 */
DWORD biSizeImage; /* 图像的像素阵列大小 */
LONG biXPelsPerMeter;
LONG biYPelsPerMeter;
DWORD biClrUsed;
DWORD biClrImportant;
} BITMAPINFOHEADER, FAR *LPBITMAPINFOHEADER, *PBITMAPINFOHEADER;
/* 调色板格式:OS/2 */
typedef struct tagRGBTRIPLE {
BYTE rgbtBlue;
BYTE rgbtGreen;
BYTE rgbtRed;
} RGBTRIPLE;
/* 调色板格式:其它格式 */
typedef struct tagRGBQUAD {
BYTE rgbBlue;
BYTE rgbGreen;
BYTE rgbRed;
BYTE rgbReserved;
} RGBQUAD;
typedef RGBQUAD FAR* LPRGBQUAD;
/* Windows GDI定义了将信息头和颜色表组合在一起的数据结构 */
typedef struct tagBITMAPINFO {
BITMAPINFOHEADER bmiHeader;
RGBQUAD bmiColors[1];
} BITMAPINFO, FAR *LPBITMAPINFO, *PBITMAPINFO;
/* Windows GDI定义了将信息头和颜色表组合在一起的数据结构 */
typedef struct tagBITMAPCOREINFO {
BITMAPCOREHEADER bmciHeader;
RGBTRIPLE bmciColors[1];
} BITMAPCOREINFO , FAR *LPBITMAPCOREINFO, *PBITMAPCOREINFO;
本文来自CSDN博客 http://blog.csdn.net/zamaolangzi/archive/2010/07/29/5774440.aspx
文章2:
位图信息头BITMAPINFOHEADER
----------------------------------------------------------------------------------------------------------
位图信息头BITMAPINFOHEADER,也是一个结构,其定义如下:
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;
这个结构的长度是固定的,为40个字节(LONG为32位整数),各个域的说明如下:
biSize
指定这个结构的长度,为40。
biWidth
指定图象的宽度,单位是象素。
biHeight
指定图象的高度,单位是象素。
biPlanes
必须是1,不用考虑。
biBitCount
指定表示颜色时要用到的位数,常用的值为1(黑白二色图), 4(16色图), 8(256色), 24(真彩色图)(新的.bmp格式支持32位色,这里就不做讨论了)。
biCompression
指定位图是否压缩,有效的值为BI_RGB,BI_RLE8,BI_RLE4,BI_BITFIELDS(都是一些Windows定义好的常量)。要说明的是,Windows位图可以采用RLE4,和RLE8的压缩格式,但用的不多。我们今后所讨论的只有第一种不压缩的情况,即biCompression为BI_RGB的情况。
biSizeImage
指定实际的位图数据占用的字节数,其实也可以从以下的公式中计算出来:
biSizeImage=biWidth’ × biHeight
要注意的是:上述公式中的biWidth’必须是4的整倍数(所以不是biWidth,而是biWidth’,表示大于或等于biWidth的,最接近4的整倍数。举个例子,如果biWidth=240,则biWidth’=240;如果biWidth=241,biWidth’=244)。
如果biCompression为BI_RGB,则该项可能为零
biXPelsPerMeter
指定目标设备的水平分辨率,单位是每米的象素个数,关于分辨率的概念。
biYPelsPerMeter
指定目标设备的垂直分辨率,单位同上。
biClrUsed
指定本图象实际用到的颜色数,如果该值为零,则用到的颜色数为2biBitCount。
biClrImportant
指定本图象中重要的颜色数,如果该值为零,则认为所有的颜色都是重要的。
文章来自于:http://hi.baidu.com/shangxing214/blog/item/4b1fec22d2eacd4cac34de6d.html
文章3:
位图文件读写综述 一、位图文件结构三、CFG_DIB的使用 位图文件由三部分组成:文件头 + 位图信息 + 位图像素数据 1、位图文件头。位图文件头主要用于识别位图文件。以下是位图文件头结构的定义: typedef struct tagBITMAPFILEHEADER { // bmfh
WORD bfType;
DWORD bfSize;
WORD bfReserved1;
WORD bfReserved2;
DWORD bfOffBits;
} BITMAPFILEHEADER; 其中的bfType值应该是“BM”(0x4d42),标志该文件是位图文件。bfSize的值是位图文件的大小。2、位图信息中所记录的值用于分配内存,设置调色板信息,读取像素值等。 以下是位图信息结构的定义:
typedef struct tagBITMAPINFO {
BITMAPINFOHEADER bmiHeader;
RGBQUAD bmiColors[1];
} BITMAPINFO;
可见位图信息也是由两部分组成的:位图信息头 + 颜色表
2.1位图信息头。位图信息头包含了单个像素所用字节数以及描述颜色的格式,此外还包括位图的宽度、高度、目标设备的位平面数、图像的压缩格式。以下是位图信息头结构的定义: typedef struct tagBITMAPINFOHEADER{ // bmih
DWORD biSize;
LONG biWidth;
LONG biHeight;
WORD biPlanes;
WORD biBitCount
DWORD biCompression;
DWORD biSizeImage;
LONG biXPelsPerMeter;
LONG biYPelsPerMeter;
DWORD biClrUsed;
DWORD biClrImportant;
} BITMAPINFOHEADER;
下表是对结构体当中各个成员的说明:
说明:*是需要加以注意的部分,因为它们是我们在进行位图操作时经常参考的变量 (2)这个值通常为0,表示使用biBitCount确定的全部颜色,例外是使用的颜色树木小于制定的颜色深度的颜色数目的最大值。 (3)这个值通常为0,表示所有的颜色都是必需的 下图是带调色板和不带调色板的位图的简单示意图 颜色表是由颜色表项组成的,颜色表项结构的定义如下: typedef struct tagRGBQUAD { // rgbq
BYTE rgbBlue;
BYTE rgbGreen;
BYTE rgbRed;
BYTE rgbReserved;
} RGBQUAD; 其中需要注意的问题是,RGBQUAD结构中的颜色顺序是BGR,而不是平常的RGB。 3、位图数据。最后,在位图文件头、位图信息头、位图颜色表之后,便是位图的主体部分:位图数据。根据不同的位图,位图数据所占据的字节数也是不同的,比如,对于8位位图,每个字节代表了一个像素,对于16位位图,每两个字节代表了一个像素,对于24位位图,每三个字节代表了一个像素,对于32位位图,每四个字节代表了一个像素。 认识了位图文件的结构以后,对特定位图文件进行读写操作就显得简单了。本文附带的源代码中包含了一个能够方便进行位图读写操作的C++类。以下给出该类的使用参考,对于实现代码中的关键部分做出了讲解。 class CFG_DIB : public CObject
{
public:
//默认构造函数
CFG_DIB();
//构造函数,根据图象宽和高,以及记录每个象素所需字节数来初始化
CFG_DIB(int width, int height, int nBitCounts);
virtual ~CFG_DIB();
public:
HBITMAP m_hBitmap;
LPBYTE m_lpDIBits; //DIB位的起始位置
LPBITMAPINFOHEADER m_lpBMPHdr; //BITMAPINFOHEADER信息
LPVOID m_lpvColorTable; //颜色表信息
HPALETTE m_hPalette; //条调色板
private:
DWORD m_dwImageSize; //非BITMAPINFOHEADER或BITMAPFILEHEADER的位
int m_nColorEntries; //颜色表项的个数
//显示参数
public:
CPoint m_Dest; //目的矩形域的左上角坐标
CSize m_DestSize; //显示矩形的宽度和高度
CPoint m_Src; //原矩形左下角坐标
CSize m_SrcSize; //原矩形宽度和高度
public:
void InitDestroy(); //初始化变量
void ComputePaletteSize(int nBitCounts); //计算调色板大小
void ComputeImage(); //计算图象大小
//从BMP文件中读入DIB信息
BOOL ReadFile(CFile* pFile);
//从BMP文件中读入DIB信息,与ReadFile不同的是使用CreateSection创建位图位
BOOL ReadSection(CFile* pFile, CDC* pDC = NULL);
//将DIB写入文件,保存成BMP图片格式
BOOL WriteFile(CFile* pFile);
//创建新的位图文件,根据参数width,height,nBitCounts分配内存空间
BOOL NewFile(int width, int height, int nBitCounts);
//关闭位图文件
BOOL CloseFile();
//显示位图
BOOL Display(CDC* pDC);
HBITMAP CreateBitmap(CDC* pDC); //用DIB创建DDB
HBITMAP CreateSection(CDC* pDC = NULL); //创建位图位数据,即象素数据
//如果DIB没有颜色表,可以用逻辑调色板
BOOL SetLogPalette(CDC* pDC);
//如果DIB有颜色表,可以创建系统调色板
BOOL SetWinPalette();
//把DIB对象的逻辑调色板选进设备环境里,然后实现调色板
UINT UseLogPalette(CDC* pDC);
//得到BitmapInfoHeader的大小,包含颜色表数据
int GetHeaderSize()
{
return sizeof(BITMAPINFOHEADER) + sizeof(RGBQUAD) * m_nColorEntries;
}
//得到图像的高度
int GetHeight()
{
if(m_lpBMPHdr == NULL) return 0;
return m_lpBMPHdr->biHeight;
}
//得到图像的宽度
int GetWidth()
{
if(m_lpBMPHdr == NULL) return 0;
return m_lpBMPHdr->biWidth;
}
//得到图像的大小
int GetImageSize()
{
return m_dwImageSize;
}
long GetLineBit(); //得到一行的象素数
}; 2、位图的读取。CFG_DIB提供了两个从位图文件读取位图数据的方法:ReadFile和ReadSection,二者不同之处,前者使用动态分配内存的方法初始化存储位位图数据的指针,后者则使用API函数,根据位图信息初始化存储位图数据的指针。 方法1 m_lpDIBits = (LPBYTE) new char[m_dwImageSize]; 方法2 m_hBitmap = ::CreateDIBSection(pDC->GetSafeHdc(),
(LPBITMAPINFO) m_lpBMPHdr, DIB_RGB_COLORS,
(LPVOID*) &m_lpDIBits, NULL, 0); 3、位图读取过程中的调色板的创建和调用。关于调色板的详细情况,本文不作详细介绍,只是对读取位图的过程中需要调用的对调色板进行操作的相关函数进行说明。 读取文件的过程中,计算出调色板大小,然后调用创建调色板函数: ComputePaletteSize(m_lpBMPHdr->biBitCount);
SetWinPalette(); 在显示位图之前,设置调色板: if(m_hPalette != NULL) {
::SelectPalette(pDC->GetSafeHdc(), m_hPalette, TRUE);
} 4、位图的显示。位图的显示还是调用Windows的API函数来进行,需要传递的参数包括当前位图信息头,位图数据等: ::StretchDIBits(pDC->GetSafeHdc(), m_Dest.x, m_Dest.y,
m_DestSize.cx, m_DestSize.cy,
m_Src.x, m_Src.y,
m_SrcSize.cx, m_SrcSize.cy,
m_lpDIBits, (LPBITMAPINFO) m_lpBMPHdr,
DIB_RGB_COLORS, SRCCOPY); 其中的m_Dest,m_DestSize,m_Src,m_SrcSize分别代表了图像在当前设备上显示的左上角坐标和范围以及需要显示的源图像的左下角坐标和范围。此处需要说明的是,位图数据的字节数组是从图像的最下面一行开始逐行想上存储的,所以用户在选取源位图的现实范围的时候需要特别注意!m_Dest,m_DestSize,m_Src,m_SrcSize需要在现实之前设置好。 5、位图的存储。位图的存储用WriteFile实现。 6、新位图的创建。新位图的创建由NewFile实现。需要的参数是位图的宽度、高度、以及位图像素占用的位数。 7、其它问题。存取位图数据的字节数组有个问题需要引起开发人员的注意:字节数组中每个扫描行的字节数必需是4的倍数,如果不足要用0补齐。 以下是处理的办法: DWORD dwBytes = ((DWORD) m_lpBMPHdr->biWidth * m_lpBMPHdr->biBitCount) / 32;
if(((DWORD) m_lpBMPHdr->biWidth * m_lpBMPHdr->biBitCount) % 32) {
dwBytes++;
}
dwBytes *= 4;
m_dwImageSize = dwBytes * m_lpBMPHdr->biHeight; 这段代码按照要求算出了用于记录图像数据的字节数组的大小。三、CFG_DIB的使用 以下是CFG_DIB的使用示例代码。 #include "fg_dib.h"
CFG_DIB m_fgdib;
//new file
m_fgdib.NewFile(width, height, nbitnum);
//open file
CFile* pf;
pf = new CFile;
pf->Open(sFileName, CFile::modeRead);
m_fgdib.ReadFile(pf);
pf->Close();
delete pf;
//draw BMP
m_fgdib.m_Dest.x = 0;
m_fgdib.m_Dest.y = 0;
m_fgdib.m_DestSize.cx = m_fgdib.GetWidth();
m_fgdib.m_DestSize.cy = m_fgdib.GetHeight();
m_fgdib.m_Src.x = 0;
m_fgdib.m_Src.y = 0;
m_fgdib.m_SrcSize.cx = m_fgdib.GetWidth();
m_fgdib.m_SrcSize.cy = m_fgdib.GetHeight();
CDC* pDC = GetDC();
m_fgdib.Display(pDC);
//close BMP
m_fgdib.CloseFile(); |
http://www.vckbase.com/document/viewdoc/?id=674