C语言读取BMP文件信息读取并在CMD窗口显示图片
位图的基本格式可以参考其它博客,这里不进行详细说明。该实验可以更加深入地理解位图的文件结构,以及C语言文件操作的方法。并且理解字节对齐的根本含义。
原始图片及文件信息
原始图片可以在网上下载24位颜色的位图,然后用画图工具进行裁剪编辑,为简单起见,我们限定只使用黑白两种颜色,位图的大小建议不要过大,因为CMD窗口无法显示完全。
效果图
字节对齐
不使用单字节对齐系统默认四字节对齐。
代码及说明
#include <stdio.h>
#include <stdlib.h>
#include <malloc.h>
//单字节对齐
#pragma pack(1)
typedef struct BitmapFileHeader
{
unsigned char bfType[2];//文件格式
unsigned long bfSize;//文件大小
unsigned short bfReserved1;//保留
unsigned short bfReserved2;//保留
unsigned long bfOffBits; //图片数据在文件中的偏移量
}fileHeader;
typedef struct BitmapInfoHeader
{
unsigned long biSize;//该结构的大小
long biWidth;//文件宽度
long biHeight;//文件高度
unsigned short biPlanes;//平面数
unsigned short biBitCount;//颜色位数
unsigned long biCompression;//压缩类型
unsigned long biSizeImage;//DIB数据区大小
long biXPixPerMeter;
long biYPixPerMeter;
unsigned long biClrUsed;//颜色索引表数目
unsigned long biClrImporant;//重要颜色数目
}fileInfo;
int main(void)
{
FILE *fpaa;
if ((fpaa = fopen("aa.bmp", "rb") )== NULL)
{
printf("打开文件失败");
exit(0);
}
printf("fpaa偏移量: %d\n", ftell(fpaa));
fileHeader *fh;
printf("fileHeader结构体大小: %d字节\n", sizeof(fileHeader));
//如果不采样单字节对齐,申请内存时相对于aa.bmp就多出两个字节
fh= (fileHeader *)malloc(sizeof(fileHeader));
fread(fh, sizeof(fileHeader), 1, fpaa);
printf("头文件格式: %c%cP\n", fh->bfType[0], fh->bfType[1]);
printf("头文件大小: %d字节\n", fh->bfSize);
printf("DIB数据在文件中的偏移量: %d字节\n", fh->bfOffBits);
printf("fpaa读取头文件后偏移量: %d\n",ftell(fpaa));
fileInfo * fi;
fi = (fileInfo *)malloc(sizeof(fileInfo));
fread(fi, sizeof(fileInfo), 1, fpaa);
printf("fpaa读取信息文件后偏移量: %d\n", ftell(fpaa));
printf("位图信息大小: %d\n",fi->biSize);
printf("文件宽度: %d\n", fi->biWidth);
printf("文件高度: %d\n", fi->biHeight);
printf("平面数: %d\n", fi->biPlanes);
printf("颜色位数: %d\n", fi->biBitCount);
printf("压缩类型: %d\n", fi->biCompression);
printf("DIB数据区大小: %d\n", fi->biCompression);
printf("颜色索引表: %d\n", fi->biClrUsed);
printf("重要颜色: %d\n", fi->biClrImporant);
printf("\n");
struct color
{
char r;
char g;
char b;
};
struct color *fc;
fc = (struct color *)malloc(sizeof(struct color));
for (int i = 1; i <=fi->biHeight; i ++)
{
fseek(fpaa, -((((fi->biBitCount * fi->biWidth) + 31) >> 5) << 2)*i, SEEK_END);
for (int j = 1; j <= fi->biWidth; j ++ )
{
fread(fc, sizeof(struct color), 1, fpaa);
if (fc->r == 0 & fc->g == 0 & fc->b == 0) printf("%c ",3);
else printf("%c ", 46);
}
printf("\n");
//((图像宽度*每个像素bites+31)/32)*4可保证每一列为4字节对齐,因为计算机为了速度每次取四字节,不足的要填0
//下方表达式显示每行多出几个字节
int k = ((((fi->biBitCount * fi->biWidth) + 31) >> 5) << 2) - ((fi->biBitCount * fi->biWidth) >> 3);
//再将文件指针移出这几个填充字节
fseek(fpaa, k, SEEK_CUR);
}
//可以用不同符号代替黑白
/*int m=10;
for (int l=1;l<100;l++)
{
printf("%d:%c ", l,l);
if (m==0)
{
m = 10;
printf("\n");
}
m--;
}
*/
getchar();
return 0;
}