C++ 文件二进制输入输出

文本文件和二进制文件

我们一般将文件分为文本文件和二进制文件。文本文件就是可以通过文本编辑器打开和编辑的文件,如使用Windows中的记事本(NotePad)或UNIX中的vi可以处理的文件。非文本文件都是二进制文件,我们不能直接使用文本编辑器编辑他们,而是通过计算机程序来处理的。常见的二进制文件,比如:图片,视频,音频以及可执行文件。

也就是说,文本文件由字符序列构成,而二进制文件由二进制位序列构成。

但是从本质上说,文件都是二进制文件。因为对于计算机来说,它只能够识别二进制的文件。我们所说的文本文件,其实本质也是二进制文件,只是通过编码(encode)将二进制位转换为了字符(编码格式想必不用多说,常见的有ASCII,GBK,Unicode等),以便人类更好地阅读。

因此我们说,文本I/O是建立在二进制I/O的基础上的,在其之上提供了一层字符编码/解码的抽象。

二进制I/O不许要任何转换。如果采用二进制I/O方式向文件中写入一个数值,那么内存中存储的值会原样复制到文件中。

那么我们如何使用二进制I/O呢?在前边的博客(C++ fstream和文件打开模式)提到过,要使用ios::binary打开文件。缺省条件下,文件是以文本文件打开的。

二进制文件的输入和输出

我们之前使用流操作符或者一些函数(put,get,getline)来读写文本文件。但是对于二进制文件,需要使用read和write函数,或者get、put、getline函数进行读写操作。

(为了方便下文用.dat文件表示二进制文件)

write函数

语法:

streamObject.write(const char* s, int size)

streamObject是流对象

s是一个字符数组

size是s的长度

下面看一个例子:

#include <iostream>
#include <fstream>
#include <string>

using namespace std;

int main()
{
    ofstream binaryio("city.dat", ios::binary);
    string city = "北京";

    binaryio.write(city.c_str(), sizeof(city));
    binaryio.close();

    cout << "二进制文件city.dat创建成功!" << endl;

    return 0;
}

运行结果:

 上面的例子中,写入的是字符数据,但是很多情况下,我们要写入非字符数据,如数字。那么我们可以使用reinterpret_cast来实现。此运算符可以将一个指针类型转换为不相关的指针类型。它只是简单地进行了指针值的二进制复制,并不改变指针所指向的数据。语法如下:

reinterpret_cast<datatype*>(address)

address是输出数据的起始地址

datatype是希望转换为的类型

在进行二进制I/O时,应转换为char类型。

看下面的例子:

#include <iostream>
#include <fstream>
#include <string>

using namespace std;

int main()
{
    ofstream binaryio("temp.dat", ios::binary);
    int value = 120;

    binaryio.write(reinterpret_cast<char*>(&value), sizeof(value));
    binaryio.close();

    cout << "二进制文件temp.dat创建成功!" << endl;

    return 0;
}

运行结果:

 

 

 

 read函数

语法:

streamObject.read(const char* address, int size)

size指最大字节数,实际读取的字符数可以从成员函数gcount中获取。

#include <iostream>
#include <fstream>
#include <string>

using namespace std;

int main()
{
    ifstream binaryio("city.dat", ios::binary);
    char s[10];

    binaryio.read(s,10);
    cout << "读入的字节数为:" << binaryio.gcount() << endl;
    s[binaryio.gcount()] = '\0';
    cout << s << endl;
    binaryio.close();

    return 0;
}

运行结果:

 

 如果想读入非字符数据,那么仍然使用reinterpret_cast

#include <iostream>
#include <fstream>
#include <string>

using namespace std;

int main()
{
    ifstream binaryio("temp.dat", ios::binary);
    int value;

    binaryio.read(reinterpret_cast<char*>(&value),sizeof(value));
    cout << "读入的字节数为:" << binaryio.gcount() << endl;
    cout << value << endl;
    binaryio.close();

    return 0;
}

运行结果:

 

posted @ 2020-05-21 13:21  川尘  阅读(6979)  评论(0编辑  收藏  举报
`