C++读取二进制文件

mrN*_*ody 3 c++ binary file

我想了解如何在 C++ 中读取二进制文件。\n我的代码:

\n\n
int main() {\n    ifstream ifd("input.png",ios::binary |ios::ate);\n    int size = ifd.tellg();\n    ifd.seekg(0,  ios::beg);\n    vector<char> buffer;\n    buffer.reserve(size);\n    ifd.read(buffer.data(), size);\n\n    cout << buffer.data();\n    return 0;\n}\n
Run Code Online (Sandbox Code Playgroud)\n\n

我认为如果我计算缓冲区,我会得到二进制结果,但事实并非如此。

\n\n

My output is: \xcb\x99\xc5\x98\xcb\x99\xc3\xa16Exif

\n\n

如果我读取文本文件,它会以正常形式而不是二进制形式显示文本。显然我的逻辑不正确。\n我如何将文件读取到缓冲区,以便它包含二进制值?\nP.s. 我这样做是为了实现 Shannon-Fano 算法,因此如果有人对读取二进制文件有任何建议,我将不胜感激。

\n

Pav*_*l P 5

您需要调整向量的大小,而不是保留它:

int main()
{
    ifstream ifd("input.png", ios::binary | ios::ate);
    int size = ifd.tellg();
    ifd.seekg(0, ios::beg);
    vector<char> buffer;
    buffer.resize(size); // << resize not reserve
    ifd.read(buffer.data(), size);

    cout.write(buffer.data(), buffer.size()); // you cannot just output buffer to cout as the buffer won't have '\0' ond-of-string terminator
}
Run Code Online (Sandbox Code Playgroud)

否则,您的代码会尝试将size字符读入空缓冲区。您也可以使用向量构造函数来设置向量大小:vector<char> buffer(size);

您可以通过这种方式输出缓冲区的字节值:

void dumpbytes(const vector<char>& v)
{
    for (int i=0; i<v.size(); ++i)
    {
        printf("%u ", (unsigned char)v[i]);
        if ((i+1) % 16 == 0)
            printf("\n");
    }
    printf("\n");
}
Run Code Online (Sandbox Code Playgroud)

或者类似于常见的十六进制编辑器对十六进制输出所做的操作:

void dumphex(const vector<char>& v)
{
    const int N = 16;
    const char hex[] = "0123456789ABCDEF";
    char buf[N*4+5+2];
    for (int i = 0; i < v.size(); ++i)
    {
        int n = i % N;
        if (n == 0)
        {
            if (i)
                puts(buf);
            memset(buf, 0x20, sizeof(buf));
            buf[sizeof(buf) - 2] = '\n';
            buf[sizeof(buf) - 1] = '\0';
        }
        unsigned char c = (unsigned char)v[i];
        buf[n*3+0] = hex[c / 16];
        buf[n*3+1] = hex[c % 16];
        buf[3*N+5+n] = (c>=' ' && c<='~') ? c : '.';
    }
    puts(buf);
}
Run Code Online (Sandbox Code Playgroud)

缓冲“Hello World!” 数据将打印如下:

48 65 6C 6C 6F 20 57 6F 72 6C 64 21                  Hello World!
Run Code Online (Sandbox Code Playgroud)