2017-04-04 47 views
1

我有一个std::fstream,我使用std :: fstream :: tellg()不正确地输出文件游标指针?

std::fstream myFile { "C:/path/to/file.txt" }; 

当我想读的第一个字节进口的,我用

char c; 
cout << myFile.tellg() << endl; // Correctly outputs 0 (begining of file) 
myFile.read(&c, 1); 
cout << myFile.tellg() << endl; // Should output 1, but it outputs 
           // FFFFFFFFFFFFFFFA 
myFile.read(&c, 1); 
cout << myFile.tellg() << endl; // Should output 2, but it outputs 
           // FFFFFFFFFFFFFFFB 

这里发生了什么?

我试图把

midi_file.seekg(0, ios_base::beg); 

midi_file.seekg(0, myFile.beg); 

但光标移动到FFFFFFFFFFFFFFFA每当我试图读取一个字节。

编辑:

我不知道这是否有事情做,但我做了一个字节序的测试,这些都是结果:

bool endianness = *reinterpret_cast<short*>("10") & 1; // Outputs 1 

编辑2:

该文件已损坏,因为输出与其他文件不同,但为什么?

这里是从文件中的字节数据,从HXD取,这是一个.midi文件:

4D 54 68 64 00 00 00 06 00 01 00 03 00 04 4D 54 
72 6B 00 00 00 A1 00 C0 69 00 90 3C 5A 01 41 5A 
01 45 5A 01 48 5A 01 49 5A 01 48 5A 01 45 5A 01 
41 5A 01 3C 5A 01 37 5A 01 33 5A 01 30 5A 01 30 
5A 01 30 5A 01 33 5A 01 37 5A 01 3C 5A 01 41 5A 
01 45 5A 01 48 5A 01 49 5A 01 48 5A 01 45 5A 01 
41 5A 01 3C 5A 01 37 5A 01 33 5A 01 30 5A 01 30 
5A 01 30 5A 01 33 5A 01 37 5A 01 3C 5A 01 41 5A 
01 45 5A 01 48 5A 01 49 5A 01 48 5A 01 45 5A 01 
41 5A 01 3C 5A 01 37 5A 01 33 5A 01 30 5A 01 30 
5A 01 30 5A 01 33 5A 01 37 5A 01 3C 5A 01 41 5A 
01 45 00 00 FF 2F 00 4D 54 72 6B 00 00 00 41 00 
C1 72 05 91 3C 5A 00 40 5A 00 43 5A 00 48 5A 0A 
35 5A 00 41 5A 00 44 5A 00 49 5A 0A 37 5A 00 40 
5A 00 43 5A 00 48 5A 0A 41 5A 00 47 5A 0A 30 5A 
00 40 5A 00 43 5A 00 48 5A 05 32 00 00 FF 2F 00 
4D 54 72 6B 00 00 00 26 00 C2 47 0A 92 50 64 01 
52 64 09 50 78 00 52 78 0A 50 00 01 52 00 09 50 
78 01 50 00 0A 52 00 00 50 00 00 FF 2F 00 

EDIT 3:

下面是该测试的全码:

#include <fstream> 
#include <iostream> 

int main() { 
    cout << std::hex << std::setfill('0') << std::uppercase; 

    fstream midi_file{ "D:/Descargas/OutFile.midi" }; 

    cout << midi_file.good() << endl; // Outputs 1 

    char c; 
    cout << midi_file.tellg() << endl; // Correctly outputs 0 (begining of file) 
    midi_file.read(&c, 1); 
    cout << midi_file.tellg() << endl; // Erroneously outputs FFFFFFFFFFFFFFFA 
    midi_file.read(&c, 1); 
    cout << midi_file.tellg() << endl; // Erroneously outputs FFFFFFFFFFFFFFFB 

    // Endianness test: 
    cout << (*reinterpret_cast<short*>("10") & 1) << endl; // Outputs 1 

    return 0; 
} 
+0

'myFile'在读取之前和之后都处于良好状态? – Angew

+0

@Angew Hmm,'myFile.badbit'返回'4','myFile.failbit'返回'2'。但是'myFile.bad()'和'myFile.fail()'返回'0'。 – Garmekain

+0

我认为当构造'std :: fstream'时,花括号'{}'是一个错字?应该是'()'。它甚至用'{}'编译? – freakish

回答

0

文件打开方式必须是ios::binary模式没有赶上特殊字符。

1

返回值为tellg的是不是的一个数字。这是一个pos_type,它有一些规则需要遵循,但在印刷不是其中之一时可以理解。请参阅http://en.cppreference.com/w/cpp/io/fpos

它的主要目的是允许查找操作返回到保存的位置。

此外,你的“排序测试”是非常非常混乱。将字符串重新解释为简短字符? C不会那样工作。 也许如果你曾经使用"\x01\x00"

+0

这是一个字符串,它只是一个'const char *',并且由于char字符串长度为2个字节,所以它可以完美地转换为2个字节的变量,在本例中为'short'。我知道''0'是十进制的'48',所以'&1'部分告诉我们这个字符的数字是否是奇数。 “0”不是“1”,所以如果“0”是MSB,则“(0x100 * 48 + 49)&1”将返回true,但如果“1”是MSB,那么'(0x100 * 49 + 48)&1'将返回false。 – Garmekain

+0

@Garmekain:如果你喜欢这样做,我猜。但它让你看起来像是一个懵懂无知的人,他很幸运。 –

+1

@Garmekain“它可以完美铸造”不,它不能。它甚至不应该编译,因为你不能用'reinterpret_cast'抛弃'const',并且当你的编译器给你一些松弛并且允许它编译时,它可以在运行时由于错误对齐而崩溃,或者编译器可以扔掉你的整个程序,因为它展示UB并用NOP替换它。 –