C ++从文件中读取并比较幻数

C++ read and compare magic number from file

我有我想用 C++ 阅读的文件。我必须阅读和检查的第一件事是文件的幻数。在我的例子中,它是十六进制值:0xABCDEF00

我是这样阅读和比较数字的:

ifstream input ("C:/Desktop/myfile", ios::binary);
if (input.is_open()) {
input.seekg(0, ios::beg);
unsigned char magic[4] = {0};
input.read((char*)magic, sizeof(magic));

if(magic[0] == 0xAB &&
   magic[1] == 0xCD &&
   magic[2] == 0xEF &&
   magic[3] == 0x00) {
   cout << "It's my File!" << endl;
} else {
   cout << "Unknown File!" << endl;
}
}

这很有效,但是有没有办法一次性比较整个读取的 char[]-Array?像这样:

unsigned int magicNumber = 0xABCDEF00;
... same code for reading file as above ...
Instead of checking each Array-Entry a way like this: 

if(magic == magicNumber) {
    do something ...
}

很高兴知道是否有这样的方法 - 如果没有,谢谢你告诉我没有这样的方法:)

如果您知道平台的字节顺序,则可以使用 uint32_t 变量来做到这一点。

对于小端系统,使用:

uint32_t number;
input.read(reinpterpret_cast<char*>(&number), 4);
if ( number == 0x00EFCDAB )
{
   cout << "It's my File!" << endl;
}

对于大端系统,使用:

uint32_t number;
input.read(reinpterpret_cast<char*>(&number), 4);
if ( number == 0xABCDEF00 )
{
   cout << "It's my File!" << endl;
}

老好人 memcmp 可以帮上忙。阅读 unsigned char magic[4] 后,您可以像这样简单地进行比较:

const unsigned char magicref[4] = {0xAB, 0xCD, 0xEF, 0}
if (memcmp(magic, magicref, sizeof(magic)) == 0) {
    // do something ...
}

这是独立于字节顺序的。

如果您知道您的平台将为您提供什么幻数并且不关心其他平台上的可移植性,您可以直接将所有内容处理为 uint32_t:

uint32_t magic, refmagic = 0xABCDEF00;  // big endian here...
input.read(reinterpret_cast<char *>(&magic), sizeof(magic)); // directly load bytes into uint32_t
if (magic == refmagic) {
    //do something...
}

这不能跨不同平台移植,但可以在简单的情况下使用,前提是用闪烁的红色粗体注释说 注意:仅在大端系统上使用

这里已经有很好的答案了!作为记录,这里有一个使用标准 <algorithm> 库的 equal() 的变体:

unsigned char magic[4] = {0};
input.read((char*)magic, sizeof(magic));

const unsigned char code[4] = { 0xab, 0xcd, 0xef, 0x00 };
if(equal(code, code+sizeof(code), magic)) 
    cout << "It's my File!" << endl;
else 
   cout << "Unknown File!" << endl;

它与 memcmp() 版本非常相似,但它适用于任何容器,而不仅仅是 char 数组。

Online demo

你可以这样做:

union magic_t {
    uint8_t bytes[4];
    uint32_t number;
};

然后如你所愿:

magic_t my_magic = {0xAB, 0xCD, 0xEF, 0};
magic_t file_magic;
input.read((char *) file_magic.bytes, sizeof(file_magic));
if ( file_magic.number == my_magic.number )...

而且您根本不需要关心字节顺序。

根据字节顺序,数字可能会有所不同,但这根本不重要,因为即使数字不是 0xABCDEF00,它也始终是正确的字节序列。

或者,可选地,我们可以只使用转换(但我认为那很丑陋)。