2

我发现自己正在编写一个简单的程序来从 bmp 文件中提取数据。我刚开始,我正处于 WTF 时刻之一。

当我运行程序并提供此图像时:http ://www.hack4fun.org/h4f/sites/default/files/bindump/lena.bmp

我得到输出:

type: 19778
size: 12
res1: 0
res2: 54
offset: 2621440

实际图像大小为 786,486 字节。为什么我的代码报告 12 个字节?

http://en.wikipedia.org/wiki/BMP_file_format中指定的标题格式 与我的 BMP_FILE_HEADER 结构相匹配。那么为什么它会充满错误的信息呢?

图像文件似乎没有损坏,并且其他图像给出同样错误的输出。我错过了什么?

#include <stdio.h>
#include <stdlib.h>

typedef struct {
    unsigned short type;
    unsigned int size;
    unsigned short res1;
    unsigned short res2;
    unsigned int offset;
} BMP_FILE_HEADER;

int main (int args, char ** argv) {
    char *file_name = argv[1];

    FILE *fp = fopen(file_name, "rb");

    BMP_FILE_HEADER file_header;

    fread(&file_header, sizeof(BMP_FILE_HEADER), 1, fp);

    if (file_header.type != 'MB') {
        printf("ERROR: not a .bmp");
        return 1;
    }

    printf("type: %i\nsize: %i\nres1: %i\nres2: %i\noffset: %i\n", file_header.type, file_header.size, file_header.res1, file_header.res2, file_header.offset);
    fclose(fp);

    return 0;
}
4

2 回答 2

2

这里是十六进制的标题:

0000000 42 4d 36 00 0c 00 00 00 00 00 36 00 00 00 28 00
0000020 00 00 00 02 00 00 00 02 00 00 01 00 18 00 00 00

长度字段是字节 36 00 0c 00`,按 intel 顺序排列;作为 32 位值处理,它是 0x000c0036 或十进制 786,486(与保存的文件大小匹配)。

可能您的 C 编译器正在将每个字段对齐到 32 位边界。启用包结构选项、编译指示或指令。

于 2012-11-13T02:43:08.143 回答
1

我可以在您的代码中找到两个错误。

第一个错误:您必须将结构打包为 1,因此每种类型的大小都是其应有的大小,因此编译器不会将其对齐,例如 4 字节对齐。所以在你的代码中short,它不是 2 个字节,而是 4 个字节。诀窍是使用编译器指令来打包最近的结构:

#pragma pack(1)

typedef struct {
    unsigned short type;
    unsigned int size;
    unsigned short res1;
    unsigned short res2;
    unsigned int offset;
} BMP_FILE_HEADER;

现在它应该正确对齐。

另一个错误在这里:

if (file_header.type != 'MB')

您正在尝试使用 1 个字节的类型(使用)检查short2 个字节的类型。可能编译器会就此向您发出警告,单引号仅包含 1 个 1 字节大小的字符是规范的。char''

为了解决这个问题,您可以将这 2 个字节分成 2 个 1 字节字符,它们是已知的 (MB),然后将它们组合成一个word. 例如:

if (file_header.type != (('M' << 8) | 'B'))

如果你看到这个表达式,就会发生这种情况:

'M'0x4D在 ASCII 中)向左移动 8 位,将导致0x4D00,现在您可以下一个字符添加到右边的零:(0x4D00 | 0x42 = 0x4D42在ASCII 中的位置)0x42'B'这样想,你可以写:

if (file_header.type != 0x4D42)

然后你的代码应该可以工作。

于 2012-11-13T02:48:26.450 回答