您的位置:首页 > 其它

256色无压缩BMP文件格式

2004-12-25 15:29 351 查看
256色的BMP文件分为 BMP文件头,BMP信息头,彩色表和位图信息矩阵4部分。

BMP文件头结构;

struct BITMAPFILEHEADER_
{
short type;//---------文件类型,一定是‘BM’
int bfSize;//---------文件大小,字节单位
short re1,re2;//------保留位
int Offbits;//--------位图矩阵偏移量,是相对于文件开头的偏移量,字节单位
};

接下来是BMP信息头

struct BITMAPINFO_
{
long size;//---------------位图大小,不一定有效的。
long width,height;//-------位图宽度和位图高度,象素单位
short planes,bitCount;//---平面数,一定为1;色彩深度,可以是1,4,8,16,分别表示单色,16色,256色和16位色。
long comp,sizeImg;//-------压缩方式,0表示无压缩,1表示RLE压缩,2表示每个象素4比特的RLE压缩。
long xpels,ypels;//--------水平分辨率和垂直分辨率,象素/米 表示
long used,important;//-----所实际使用的颜色表中的颜色数,不一定有效;重要的颜色数,也不一定有效
};

彩色表项的结构是
struct COLOR_
{
unsigned char blue;//--------蓝色亮度
unsigned char green;//-------绿色亮度
unsigned char red;//---------红色亮度
unsigned char re;//----------保留
}

RLE是(Run Length Encoded 游程长度编码)压缩

这里只处理256色无压缩的BMP文件。

下面是BCB中读取BMP文件并在画布中显示出来的代码。

#include <vcl.h>
#pragma hdrstop
#include<stdio.h>
#include "Unit1.h"
#include"File1.h"

#pragma pack(1)
struct BITMAPFILEHEADER_
{
short type;
int bfSize;
short re1,re2;
int Offbits;
};

struct BITMAPINFO_
{
long size;
long width,height;
short planes,bitCount;
long comp,sizeImg;
long xpels,ypels;
long used,important;
};

//-------------将BMP彩色表的数据校正到BCB TColor的数据。
void SwitchColor(long &c)
{
long blue=c& 0x000000ff;
long green=c& 0x0000ff00;
long red=c& 0x00ff0000;
c=(blue<<16) | green | (red>>16);
}

void xxx()
{
FILE *f=fopen("F://FX3.bmp","rb");
if(f==NULL) /*判断文件是否打开成功*/
{
ShowMessage("File open error");
return;
}

fseek(f,0,0);//移动到开头

//----------读BMP文件头
BITMAPFILEHEADER_ *bmph=new BITMAPFILEHEADER_();
if(fread((char*)bmph,sizeof(BITMAPFILEHEADER_),1,f)==NULL)
{
ShowMessage("File read error");
return;
}

//-----------读BMP信息头
BITMAPINFO_ *bmpi=new BITMAPINFO_();
if(fread((char*)bmpi,sizeof(BITMAPINFO_),1,f)==NULL)
{
ShowMessage("File read error2");
return;
}

//--------------读彩色表
long *c=new long[bmph->Offbits-sizeof(BITMAPFILEHEADER_)-sizeof(BITMAPINFO_)];
fread((char*)c,bmph->Offbits-sizeof(BITMAPFILEHEADER_)-sizeof(BITMAPINFO_),1,f);

//------------显示图形
unsigned char *p=new unsigned char[4];
int i=0,j=0,k=0,wc=0;
TColor *tc;
if(bmpi->width%4==0)//-----------因为BMP图像4字节对齐
wc=bmpi->width/4;
else
wc=bmpi->width/4+1;

for( i=0;i<bmpi->height;i++)
{
for(j=0;j<wc;j++)
{
fread(p,4,1,f);
for(k=0;k<4;k++)
{
long x=c[p[k]];
SwitchColor(x);//----------因为BCB的TCOLOR和BMP的彩色表反了。
Form1->Canvas->Pixels[200+j*4+k][300-i]=x; //------200和300是定位到Canvas的中间而已。
}
}
}
fclose(f);
};

BMP文件是从下倒上,从左到右倒向存储的,位图矩阵的第一行时图像的最底一行。

BMP的行象素是4字节对齐的,不足的补0,比如有个图像每行宽度是63象素,BMP文件存储时会每行存储64个字节,最后一个字节用0补齐。BMP信息头中的width是实际的行象素数,比如这里会是63,显示时读到每行第63个字节时要再读一个补齐的字节,然后才能换行。

BCB中写点函数用的TColor和BMP文件的彩色表的字节顺序有出入。
TColor的结构是0x00bbggrr,rr是8位红色亮度,gg是8位绿色亮度,bb是8位红色亮度,最高8位保留为0;
而BMP彩色表项的结构是0x00rrggbb,红色和蓝色和TColor的反过来了,所以在BCB中要校正过来,我的switchColor函数实现这个功能。不过我没有用COLOR_结构,直接用了long型数据。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: