此课件及源代码来自B站up主:码农论坛,该文章仅作为本人学习笔记使用。

1、文件操作-写入二进制文件

二进制文件以数据块的形式组织数据,把内存中的数据直接写入文件。

包含头文件:#include <fstream>

类:ofstream(output file stream)

ofstream打开文件的模式(方式):

对于ofstream,不管用哪种模式打开文件,如果文件不存在,都会创建文件。

ios::out             缺省值:会截断文件内容。

ios::trunc          截断文件内容。(truncate)

ios::app            不截断文件内容,只在文件未尾追加文件。(append)

ios::binary        以二进制方式打开文件。

特别注意:写入文件时不要写入string类等类对象,其写入文件中的数据包括了动态开辟的指针,读取时会出错。

操作文本文件和二进制文件的一些细节:

1)在windows平台下,文本文件的换行标志是"\r\n"。

2)在linux平台下,文本文件的换行标志是"\n"。

3)在windows平台下,如果以文本方式打开文件,写入数据的时候,系统会将"\n"转换成"\r\n";读取数据的时候,系统会将"\r\n"转换成"\n"。 如果以二进制方式打开文件,写和读都不会进行转换。

4)在Linux平台下,以文本或二进制方式打开文件,系统不会做任何转换。

5)以文本方式读取文件的时候,遇到换行符停止,读入的内容中没有换行符;以二制方式读取文件的时候,遇到换行符不会停止,读入的内容中会包含换行符(换行符被视为数据)。

6)在实际开发中,从兼容和语义考虑,一般:a)以文本模式打开文本文件,用行的方法操作它;b)以二进制模式打开二进制文件,用数据块的方法操作它;c)以二进制模式打开文本文件和二进制文件,用数据块的方法操作它,这种情况表示不关心数据的内容。(例如复制文件和传输文件)d)不要以文本模式打开二进制文件,也不要用行的方法操作二进制文件,可能会破坏二进制数据文件的格式,也没有必要。(因为二进制文件中的某字节的取值可能是换行符,但它的意义并不是换行,可能是整数n个字节中的某个字节)

示例:

#include <iostream>
#include <fstream>  // ofstream类需要包含的头文件。
using  namespace std;

int main()
{
      // 文件名一般用全路径,书写的方法如下:
      //  1)"D:\data\bin\test.dat"       // 错误。
      //  2)R"(D:\data\bin\test.dat)"   // 原始字面量,C++11标准。
      //  3)"D:\\data\\bin\\test.dat"   // 转义字符。
      //  4)"D:/tata/bin/test.dat"        // 把斜线反着写。
      //  5)"/data/bin/test.dat"          //  Linux系统采用的方法。
      string filename = R"(D:\data\bin\test.dat)";
      //char    filename[] = R"(D:\data\bin\test.dat)";

      // 创建文件输出流对象,打开文件,如果文件不存在,则创建它。
      //ofstream fout(filename, ios::binary);
      //ofstream fout(filename, ios::out | ios::binary);
      //ofstream fout(filename, ios::trunc | ios::binary);
      //ofstream fout(filename, ios::app | ios::binary);
      ofstream fout;

      fout.open(filename, ios::app | ios::binary);

      // 判断打开文件是否成功。
      // 失败的原因主要有:1)目录不存在;2)磁盘空间已满;3)没有权限,Linux平台下很常见。
      if (fout.is_open() == false)
      {
            cout << "打开文件" << filename << "失败。\n";  return 0;
      }

      // 向文件中写入数据。
      struct Data {               // 结构体。
            char name[31];         // 姓名。
            int    no;                    // 编号。  
      }data;

      data = { "zq",3};
      fout.write((const char*)&data, sizeof(Data));   // 写入第一块数据。

      data = { "qq",8};
      fout.write((const char*)&data, sizeof(Data));     // 写入第二块数据。

      fout.close();        // 关闭文件,fout对象失效前会自动调用close()。

      cout << "操作文件完成。\n";
}

2、文件操作-读取二进制文件

包含头文件:#include <fstream>

类:ifstream

ifstream打开文件的模式(方式):

对于ifstream,如果文件不存在,则打开文件失败。

ios::in               缺省值。

ios::binary        以二进制方式打开文件。

示例:

#include <iostream>
#include <fstream>  // ifstream类需要包含的头文件。
using  namespace std;

int main()
{
      // 文件名一般用全路径,书写的方法如下:
      //  1)"D:\data\bin\test.dat"       // 错误。
      //  2)R"(D:\data\bin\test.dat)"   // 原始字面量,C++11标准。
      //  3)"D:\\data\\bin\\test.dat"   // 转义字符。
      //  4)"D:/tata/bin/test.dat"        // 把斜线反着写。
      //  5)"/data/bin/test.dat"          //  Linux系统采用的方法。
      string filename = R"(D:\data\bin\test.dat)";
      //char    filename[] = R"(D:\data\bin\test.dat)";

      // 创建文件输入流对象,打开文件,如果文件不存在,则打开文件失败。。
      //ifstream fin(filename , ios::binary);
      //ifstream fin(filename , ios::in | ios::binary);
      ifstream fin;

      fin.open(filename, ios::in | ios::binary);

      // 判断打开文件是否成功。
      // 失败的原因主要有:1)目录不存在;2)文件不存在;3)没有权限,Linux平台下很常见。
      if (fin.is_open() == false)
      {
            cout << "打开文件" << filename << "失败。\n";  return 0;
      }

      // 二进制文件以数据块(数据类型)的形式组织数据。
      struct Data {               // 结构体。
            char name[31];         // 姓名。
            int    no;                    // 编号。  
      }data;

      while (fin.read((char*)&data, sizeof(data)))
      {
            cout << "name=" << girl.name << ",no=" << girl.no << endl;
      }

      fin.close();     // 关闭文件,fin对象失效前会自动调用close()。

      cout << "操作文件完成。\n";
}