BMP图像格式学习

对应数据结构

1:BMP文件组成
BMP文件由 文件头、位图信息头、颜色信息和 图形数据四部分组成。
2:BMP文件头(14字节)
BMP文件头数据结构含有BMP文件的类型、文件大小和位图起始位置等信息。
其结构定义如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
typedef struct tagBITMAPFILEHEADER
{
    WORD bfType; // 位图文件的类型,必须为BM(1-2字节)
    DWORD bfSize; // 位图文件的大小,以字节为单位(3-6字节)
    WORD bfReserved1; // 位图文件保留字,必须为0(7-8字节)
    WORD bfReserved2; // 位图文件保留字,必须为0(9-10字节)
    DWORD bfOffBits; // 位图数据的起始位置,以相对于位图(11-14字节)
    // 文件头的偏移量表示,以字节为单位
} BITMAPFILEHEADER;
3:位图信息头(40字节)
BMP位图信息头数据用于说明位图的尺寸等信息。
typedef struct tagBITMAPINFOHEADER{
    DWORD biSize; // 本结构所占用字节数(15-18字节)
    LONG biWidth; // 位图的宽度,以像素为单位(19-22字节)
    LONG biHeight; // 位图的高度,以像素为单位(23-26字节)
    WORD biPlanes; // 目标设备的级别,必须为1(27-28字节)
    WORD biBitCount; // 每个像素所需的位数,必须是1(双色),(29-30字节)
    // 4(16色),8(256色)或24(真彩色)之一
    DWORD biCompression; // 位图压缩类型,必须是 0(不压缩),(31-34字节)
    // 1(BI_RLE8压缩类型)或2(BI_RLE4压缩类型)之一
    DWORD biSizeImage; // 位图的大小,以字节为单位(35-38字节)
    LONG biXPelsPerMeter; // 位图水平分辨率,每米像素数(39-42字节)
    LONG biYPelsPerMeter; // 位图垂直分辨率,每米像素数(43-46字节)
    DWORD biClrUsed; // 位图实际使用的颜色表中的颜色数(47-50字节)
    DWORD biClrImportant; // 位图显示过程中重要的颜色数(51-54字节)
} BITMAPINFOHEADER;
4:颜色表
颜色表用于说明位图中的颜色,它有若干个表项,每一个表项是一个RGBQUAD类型的结构,定义一种颜色。RGBQUAD结构的定义如下:
1
2
3
4
5
6
typedef struct tagRGBQUAD {
    BYTE rgbBlue; // 蓝色的亮度(值范围为0-255)
    BYTE rgbGreen; // 绿色的亮度(值范围为0-255)
    BYTE rgbRed; // 红色的亮度(值范围为0-255)
    BYTE rgbReserved; // 保留,必须为0
} RGBQUAD;
颜色表中RGBQUAD结构数据的个数有biBitCount来确定:
当biBitCount=1,4,8时,分别有2,16,256个表项;
当biBitCount=24时,没有颜色表项。
位图信息头和颜色表组成位图信息,BITMAPINFO结构定义如下:
1
2
3
4
typedef struct tagBITMAPINFO {
    BITMAPINFOHEADER bmiHeader; // 位图信息头
    RGBQUAD bmiColors[1]; // 颜色表
} BITMAPINFO;
5:位图数据
位图 数据记录了位图的每一个像素值,记录顺序是在扫描行内是从左到右,扫描行之间是从下到上。位图的一个像素值所占的字节数:
当biBitCount=1时,8个像素占1个字节;
当biBitCount=4时,2个像素占1个字节;
当biBitCount=8时,1个像素占1个字节;
当biBitCount=24时,1个像素占3个字节;
Windows规定一个扫描行所占的字节数必须是
4的倍数(即以long为单位),不足的以0填充,
biSizeImage = ((((bi.biWidth * bi.biBitCount) + 31) & ~31) / 8) * bi.biHeight;
具体数据举例:
如某BMP文件开头:
424D 46900000 0000 0000 4600 0000 2800 0000 8000 0000 9000 0000 0100*1000 0300 0000 0090 0000 A00F 0000 A00F0000 0000 00000000 0000*00F8 E007 1F00 0000*02F1 84F1 04F1 84F1 84F1 06F2 84F1 06F2 04F2 86F2 06F2 86F2 86F2 .... ....

读取方法

/*
功能:在图片的第50行画一条黑线
为简化代码,只支持24位色的图片
codeblocks下正确运行。VC下需要将二维数组img改为malloc动态分配。
*/
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
#include <stdio.h>
#include <windows.h>
typedef struct {
    BYTE b;
    BYTE g;
    BYTE r;
}RGB;
int main( void )
{
    BITMAPFILEHEADER fileHeader;
    BITMAPINFOHEADER infoHeader;
    FILE * pfin = fopen ( "原始图像.bmp" , "rb" );
    FILE * pfout = fopen ( "修改后的图像.bmp" , "wb" );
    //Read the Bitmap file header;
    fread (&fileHeader, sizeof (BITMAPFILEHEADER),1,pfin);
    //Read the Bitmap info header;
    fread (&infoHeader, sizeof (BITMAPINFOHEADER),1,pfin);
    //为简化代码,只处理24位彩色
if ( infoHeader.biBitCount == 24 )
{
    int size = infoHeader.biWidth*infoHeader.biHeight;
    RGB img[infoHeader.biHeight][infoHeader.biWidth];
    fread ( img , sizeof (RGB) , size , pfin );
    //把第50行染成黑色
    int i = 0;
    for ( ; i < infoHeader.biWidth ; i++ )
    {
         img[50][i].b =img[50][i].g=img[50][i].r= 0;
    }
    //将修改后的图片保存到文件
    fwrite ( &fileHeader , sizeof (fileHeader) , 1 , pfout );
    fwrite ( &infoHeader , sizeof (infoHeader) , 1 , pfout );
    fwrite ( img , sizeof (RGB) , size , pfout );
}
fclose (pfin);
fclose (pfout);
}

2文件部分

图像文件头

1)1-2:(这里的数字代表的是"字",即两个字节,下同)图像文件头。0x4d42=’BM’,表示是Windows支持的 BMP格式。( 注意:查ascii表B 0x42,M0x4d,bfType 为两个字节,B为low字节,M为high字节所以bfType=0x4D42,而不是0x424D,但注意)
2)3-6:整个文件大小。4690 0000,为00009046h=36934。
3)7-8:保留,必须设置为0。
4)9-10:保留,必须设置为0。
5)11-14:从文件开始到位图数据之间的 偏移量(14+40+4*(2^biBitCount))。4600 0000,为00000046h=70,上面的文件头就是35字=70字节。

位图信息头

6)15-18:位图图信息头长度。
7) 19-22:位图宽度,以像素为单位。8000 0000,为00000080h=128。
8)23-26:位图高度,以像素为单位。9000 0000,为00000090h=144。
9)27-28:位图的位面数,该值总是1。0100,为0001h=1。
10)29-30:每个像素的位数。有1(单色),4(16色),8(256色),16(64K色,高彩色),24(16M色,真彩色),32(4096M色,增强型真彩色)。1000为0010h=16。
11)31-34:压缩说明:有0(不压缩),1(RLE 8,8位RLE压缩),2(RLE 4,4位RLE压缩,3(Bitfields,位域存放)。RLE简单地说是采用像素数+像素值的方式进行压缩。T408采用的是位域存放方式,用两个字节表示一个像素,位域分配为r5b6g5。图中0300 0000为00000003h=3。
12)35-38:用字节数表示的位图数据的大小,该数必须是4的倍数, 数值上等于(≥位图宽度的最小的4的倍数)×位图高度×每个像素位数。0090 0000为00009000h=80×90×2h=36864。
13)39-42:用 象素/米表示的 水平分辨率。A00F 0000为0000 0FA0h=4000。
14)43-46:用象素/米表示的垂直 分辨率。A00F 0000为0000 0FA0h=4000。
15)47-50:位图使用的颜色索引数。设为0的话,则说明使用所有调色板项。
16)51-54:对图象显示有重要影响的颜色索引的数目。如果是0,表示都重要。

彩色板

17)(55+0)到(50-1+2^biBitCount):彩色板规范。对于调色板中的每个表项,用下述方法来描述RGB的值:
1字节用于蓝色分量
1字节用于绿色分量
1字节用于红色分量
1字节用于填充符(设置为0)
对于24-位 真彩色图像就不使用彩色板,因为位图中的RGB值就代表了每个象素的颜色。
如,彩色板为00F8 0000 E007 0000 1F00 0000 0000 0000,其中:
00F8为F800h = 1111 1000 0000 0000(二进制),是蓝色分量的掩码。
E007 为 07E0h = 0000 0111 1110 0000(二进制),是绿色分量的掩码。
1F00为001Fh = 0000 0000 0001  [1] 1111(二进制),是红色分量的掩码。
0000 总设置为0。
将掩码跟像素值进行“与”运算再进行移位操作就可以得到各色分量值。看看掩码,就可以明白事实上在每个像素值的两个字节16位中,按从高到低取5、6、5位分别就是r、g、b分量值。取出分量值后把r、g、b值分别乘以8、4、8就可以补齐第个分量为一个字节,再把这三个字节按rgb组合,放入存储器(同样要反序),就可以转换为24位标准BMP格式了。

图像数据阵列

18)55(无调色板)-bfSize:每两个字节表示一个像素。阵列中的第一个字节表示位图左下角的 象素,而最后一个字节表示位图右上角的象素。

存储算法

BMP文件通常是不压缩的,所以它们通常比同一幅图像的压缩图像 文件格式要大很多。例如,一个800×600的24位几乎占据1.4MB空间。因此它们通常不适合在因特网或者其它低速或者有容量限制的媒介上进行传输。根据颜色深度的不同,图像上的一个像素可以用一个或者多个 字节表示,它由n/8所确定(n是位深度,1字节包含8个 数据位)。图片浏览器等基于 字节的ASCII值计算 像素的颜色,然后从调色板中读出相应的值。更为详细的信息请参阅下面关于 位图文件的部分。n位2n种颜色的位图近似字节数可以用下面的公式计算:BMP文件大小约等于 54+4*2的n次方+(w*h*n)/8,其中高度和宽度都是像素数。需要注意的是上面公式中的54是位图文件的 文件头,是彩色调色板的大小。另外需要注意的是这是一个近似值,对于n位的 位图图像来说,尽管可能有最多2n中颜色,一个特定的图像可能并不会使用这些所有的颜色。由于彩色 调色板仅仅定义了图像所用的颜色,所以实际的彩色调色板将小于。如果想知道这些值是如何得到的,请参考下面 文件格式的部分。由于存储 算法本身决定的因素,根据几个 图像参数的不同计算出的大小与实际的文件大小将会有一些细小的差别。

3数据类型

Oracle产品 EPC Bitmap 中数据类型 EPC bitmap的缩写。
Oracle提出了一种EPC bitmap的数据类型对基于RFID产品项级别跟踪应用产生的大量的数据进行有效处理,。
EPC bitmap数据类型定义一个EPC集合,集合共享EPC的一些特征(例如header,manager number,and object class)。支持这个数据类型的关键是使用RFID标识的物品项,在一个群组里可以基于共同属性(例如位置,截止日期,或制造商),在通常情况下可以被追踪。而EPC集合可以表示为一个EPC bitmap,bitmap可以被访问并可通过epc2bmp或bmp2epc进行数据类型的转换操作。
引入EPC bitmap(epc bmp)类型的好处:
(1)可以简单的标识一个RFID EPC集合,而不会丢失任何信息;
(2)对于同类的EPC操作可以简单的在bitmap上操作,简化了操作方式



测试用例图片。不要积分:http://download.csdn.net/detail/u011141103/6342001
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值