C语言读取BMP文件信息读取并在CMD窗口显示图片

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;
}

猜你喜欢

转载自blog.csdn.net/chulan3207/article/details/82118775
今日推荐