C++基础(1) - 文件操作

1. 文件的概念

        对于用户来说,常用到的文件有两大类:程序文件和数据文件。而根据文件中数据的组织方式,则可以将文件分为ASCII 文件和二进制文件。

数字 64 在内存中表示为 0100 0000,若将其保存为 ASCII 文件,则要分别存放十位 6 和个位 4 的 ASCII 码,为 0011 0110 0011 0100,占用两个字节;

若将其保存为二进制文件,则按内存中形式直接输出,为 0100 0000,占用一个字节。

ASCII 文件中数据与字符一一对应,一个字节代表一个字符,可以直接在屏幕上显示或打印出来,这种方式使用方便,比较直观,便于阅读,

但一般占用存储空间较大,而且输出时要将二进制转化为 ASCII 码比较花费时间。

二进制文件,输出时不需要进行转化,直接将内存中的形式输出到文件中,占用存储空间较小,但一个字节并不对应一个文件,不能直观显示文件中的内容。

2. 文件流和文件流对象

        文件流是以外存文件未输入输出对象的数据流。输出文件流是从内存流向外存文件的数据,输入文件流是从外存文件流向内存的数据。每一个文件

流都有一个内存缓冲区与之对应。

C++ 中有三个用于文件操作的文件类:

  • ifstream 类,它是从 istream 类派生来的,用于支持从磁盘文件的输入。
  • ofstream 类,它是从 ostream 类派生来的,用于支持向磁盘文件的输出。
  • fstream 类,它是从 iostream 类派生来的,用于支持对磁盘文件的输入输出。

要以磁盘文件为对象进行输入输出,必须定义一个文件流类的对象,通过文件流对象将数据从内存输出到磁盘文件,或者将磁盘文件输入到内存。

定义文件流对象后,我们还需要将文件流对象和指定的磁盘文件建立关联,以便使文件流流向指定的磁盘文件,并确定文件的工作方式(输入还是输出,

二进制还是 ASCII)。我们可以在定义流对象的时候指定参数来调用构造函数,或者通过成员函数 open 来进行文件流对象和指定文件的关联。

fstream 类拥有 ifstream 和 ofstream 类中所有的成员方法,表 2 罗列了 fstream 类一些常用的成员方法。

成员方法名

适用类对象

功   能

open()

fstream

ifstream

ofstream

打开指定文件,使其与文件流对象相关联。

is_open()

检查指定文件是否已打开。

close()

关闭文件,切断和文件流对象的关联。

swap()

交换 2 个文件流对象。

operator>>

fstream

ifstream

重载 >> 运算符,用于从指定文件中读取数据。

gcount()

返回上次从文件流提取出的字符个数。该函数常和 get()、getline()、ignore()、peek()、read()、readsome()、putback() 和 unget() 联用。

get()

从文件流中读取一个字符,同时该字符会从输入流中消失。

getline(str,n,ch)

从文件流中接收 n-1 个字符给 str 变量,当遇到指定 ch 字符时会停止读取,默认情况下 ch 为 '\0'。

ignore(n,ch)

从文件流中逐个提取字符,但提取出的字符被忽略,不被使用,直至提取出 n 个字符,或者当前读取的字符为 ch。

peek()

返回文件流中的第一个字符,但并不是提取该字符。

putback(c)

将字符 c 置入文件流(缓冲区)。

operator<<

fstream

ofstream

重载 << 运算符,用于向文件中写入指定数据。

put()

向指定文件流中写入单个字符。

write()

向指定文件中写入字符串。

tellp()

用于获取当前文件输出流指针的位置。

seekp()

设置输出文件输出流指针的位置。

flush()

刷新文件输出流缓冲区。

good()

fstream

ofstream

ifstream

操作成功,没有发生任何错误。

eof()

到达输入末尾或文件尾。

                                                        表2:fstream类常用成员方法

3.文件的打开

    打开文件可以通过以下两种方式进行:

  • 调用流对象的 open 成员函数打开文件。
  • 定义文件流对象时,通过构造函数打开文件。

使用 open 函数打开文件

先看第一种文件打开方式。以 ifstream 类为例,该类有一个 open 成员函数,其他两个文件流类也有同样的 open 成员函数:

void open(const char* szFileName, int mode)

第一个参数是指向文件名的指针,第二个参数是文件的打开模式标记。

文件的打开模式标记代表了文件的使用方式,这些标记可以单独使用,也可以组合使用。表 1 列出了各种模式标记单独使用时的作用,以及常见的两种模式标记组合的作用。

模式标记

适用对象

作用

ios::in

ifstream

fstream

打开文件用于读取数据。如果文件不存在,则打开出错。

ios::out

ofstream

fstream

打开文件用于写入数据。如果文件不存在,则新建该文件;如果文件原来就存在,则打开时清除原来的内容。

ios::app

ofstream

fstream

打开文件,用于在其尾部添加数据。如果文件不存在,则新建该文件。

ios::ate

ifstream

打开一个已有的文件,并将文件读指针指向文件末尾(读写指 的概念后面解释)。如果文件不存在,则打开出错。

ios:: trunc

ofstream

打开文件时会清空内部存储的所有数据,单独使用时与 ios::out 相同。

ios::binary

ifstream

ofstream

fstream

以二进制方式打开文件。若不指定此模式,则以文本模式打开。

ios::in | ios::out

fstream

打开已存在的文件,既可读取其内容,也可向其写入数据。文件刚打开时,原有内容保持不变。如果文件不存在,则打开出错。

ios::in | ios::out

ofstream

打开已存在的文件,可以向其写入数据。文件刚打开时,原有内容保持不变。如果文件不存在,则打开出错。

ios::in | ios::out | ios::trunc

fstream

打开文件,既可读取其内容,也可向其写入数据。如果文件本来就存在,则打开时清除原来的内容;如果文件不存在,则新建该文件。

                                                 表1:文件打开模式标记

ios::binary 可以和其他模式标记组合使用,例如:

  • ios::in | ios::binary表示用二进制模式,以读取的方式打开文件。
  • ios::out | ios::binary表示用二进制模式,以写入的方式打开文件。

文本方式与二进制方式打开文件的区别其实非常微小,一般来说,如果处理的是文本文件,那么用文本方式打开会方便一些。但其实任何文件都可以以二进

制方式打开来读写。

在流对象上执行 open 成员函数,给出文件名和打开模式。判断文件打开是否成功,可以看“对象名”这个表达式的值是否为 true,如果为true,则表示打开成功。

​#include<iostream>
#include<fstream>
using namespace std;

int main()
{
    
ifstream inFile;

inFile.open("c:\\tmp\\test.txt",ios::in);
if(inFile)//条件成立,则说明文件打开成功
    inFile.close();
else
    cout <<"test.txt doesn't exist"<<endl;
    
ofstream oFile;
oFile.open("test1.txt",ios::out);
if(!oFile)//条件成立,则说明文件打开出错
    cout <<"error 1"<<endl;
else
    oFile.close();

oFile.open("tmp\\test2.txt",ios::out |ios::in);
if(oFile)//条件成立,则说明文件打开成功
    oFile.close();
else
    cout <<"error 2"<<endl;

fstream ioFile;
ioFile.open("..\\test3.txt",ios::out |ios::in |ios::trunc);
if(!ioFile)
    cout <<"error 3"<<endl;
else
    ioFile.close();
return0;
}

​

使用流类的构造函数打开文件

定义流对象时,在构造函数中给出文件名和打开模式也可以打开文件。以 ifstream 类为例,它有如下构造函数:

ifstream::ifstream (const char* szFileName, int mode = ios::in, int);

第一个参数是指向文件名的指针;第二个参数是打开文件的模式标记,默认值为ios::in; 第三个参数是整型的,也有默认值,一般极少使用。

在定义流对象时打开文件的示例程序如下(用流类的构造函数打开文件):

#include<iostream>
#include<fstream>
using namespace std;
int main()
{
ifstream inFile("c:\\tmp\\test.txt",ios::in);
if(inFile)
    inFile.close();
else
    cout <<"test.txt doesn't exist"<<endl;

ofstream oFile("test1.txt",ios::out);
if(!oFile)
    cout <<"error 1";
else
    oFile.close();

fstream oFile2("tmp\\test2.txt",ios::out |ios::in);
if(!oFile2)
    cout <<"error 2";
else
    oFile.close();

return 0;

}

4. 文件的操作

对 ASCII 文件的操作

然后,我们就可以用类似 cin 或者 cout 的方式将数据读出或写入文件,只不过是输入输出的对象变成了文件而已。当然,在对磁盘文件完成读写操作后,我们可以通过 close 方法来解除磁盘文件和文件流对象的关联。

#include <iostream>
#include <fstream>
using namespace std;
int main()
{
    ofstream outfile("a.txt", ios::out);
    if (!outfile)
    {
        cerr << "Failed to open the file!";
        return 1;
    }

    // 写入数字 1-5 到文件中
    for (int i = 1; i < 6; i++)
    {
        outfile << i << '\n';
    }
    outfile.close();

    ifstream infile("a.txt", ios::in);
    if (!infile)
    {
        cerr << "Failed to open the file!";
        return 1;
    }

    char data;  // 从文件中读出数字 1-5
    for (int i = 1; i < 6; i++)
    {
        infile >> data;
        cout << data << '\n';
    }
    infile.close();
    return 0;
}

也可以利用文件流对象的成员函数 get, put 等,其用法就和标准输入输出介绍的一样

int main()
{
    ofstream outfile("a.txt", ios::out);
    if (!outfile)
    {
        cerr << "Failed to open the file!";
        return 1;
    }

    for (char i = '1'; i < '6'; i++)
    {
        outfile.put(i); // 输出一个字符到文件中去
    }
    outfile.close();

    ifstream infile("a.txt", ios::in);
    if (!infile)
    {
        cerr << "Failed to open the file!";
        return 1;
    }

    /*char a;
    for (int i = 0; i < 5; i++)
    {
        infile.get(a); // 从文件中读出 1 个字符
        cout << a << '\n';
    }*/

    char data[5];
    infile.get(data, 6); // 从文件中读出 5 个字符
    for (int i = 0; i < 5; i++)
    {
       cout << data[i] << '\n';
    }
    infile.close();
    return 0;
}

对二进制文件的操作

二进制文件的操作需要在打开文件的时候指定打开方式为 ios::binary,并且还可以指定为既能输入又能输出的文件,我们通过成员函数 read 和 write 来读写二进制文件。

  • istream& read (char* s, streamsize n);
  • ostream& write (const char* s, streamsize n);
#include <iostream>
#include <fstream>
using namespace std;
int main()
{
    ofstream outfile("a.txt", ios::binary);
    if (!outfile)
    {
        cerr << "Failed to open the file!";
        return 1;
    }

    char a[] = {'h', 'e', 'l', 'l', 'o', ','};
    char b[] = {'s', 'e', 'n', 'i', 'u', 's', 'e', 'n', '!'};
    outfile.write(a, 6); // 将以 a 为首地址的 6 个字符写入文件
    outfile.write(b, 9);
    outfile.close();

    ifstream infile("a.txt", ios::binary);
    if (!infile)
    {
        cerr << "Failed to open the file!";
        return 1;
    }

    char data[6];
    infile.read(data, 6);  // 从文件中读出 6 个字符到以 data 为首地址的字符数组中
    for (int i = 0; i < 6; i++)
    {
        cout << data[i];
    }

    char datb[6];
    infile.read(datb, 9);
    for (int i = 0; i < 9; i++)
    {
        cout << datb[i];
    }
    infile.close();
    return 0;
}

在磁盘文件中有一个文件指针,用来指明当前读写的位置。每次写入或者读出一个字节,指针就向后移动一个字节。对于二进制文件,允许对指针进行控制,使它移动到所需的位置,以便在该位置上进行读写。

  • ostream& seekp (streampos pos);将输出文件中指针移动到指定的位置
  • ostream& seekp (streamoff off, ios_base::seekdir way);以参照位置为基准对输出文件中的指针移动若干字节
  • streampos tellp();返回输出文件指针当前的位置
  • istream& seekg (streampos pos);将输入文件中指针移动到指定的位置
  • istream& seekg (streamoff off, ios_base::seekdir way);以参照位置为基准对输入文件中的指针移动若干字节
  • streampos tellg();返回输入文件指针当前的位置

其中,参照位置有以下几个选择:

  • ios_base::beg文件开始位置
  • ios_base::cur文件当前位置
  • ios_base::end文件末尾位置

getline():从文件中读取一行字符串

#include<iostream>
#include<fstream>
usingnamespacestd;
intmain()
{
charc[40];

//以二进制模式打开 in.txt 文件
ifstream inFile("in.txt",ios::in |ios::binary);

//判断文件是否正常打开
if(!inFile){
    cout <<"error"<<endl;
    return0;
}

//从 in.txt 文件中读取一行字符串,最多不超过 39 个
inFile.getline(c,40);
cout <<c ;
inFile.close();
return0;
}

移动和获取文件读写指针(seekpseekgtellgtellp

在读写文件时,有时希望直接跳到文件中的某处开始读写,这就需要先将文件的读写指针指向该处,然后再进行读写。

  • ifstream 类和 fstream 类有 seekg 成员函数,可以设置文件读指针的位置;
  • ofstream 类和 fstream 类有 seekp 成员函数,可以设置文件写指针的位置。

所谓“位置”,就是指距离文件开头有多少个字节。文件开头的位置是 0。

这两个函数的原型如下:

ostream & seekp (int offset, int mode);
istream & seekg (int offset, int mode);

seekg()是对输入流的操作 g是get缩写

seekp()是对输出流的操作 p是put缩写

mode 代表文件读写指针的设置模式,有以下三种选项:

  • ios::beg:让文件读指针(或写指针)指向从文件开始向后的 offset 字节处。offset 等于 0 即代表文件开头。在此情况下,offset 只能是非负数。
  • ios::cur:在此情况下,offset 为负数则表示将读指针(或写指针)从当前位置朝文件开头方向移动 offset 字节,为正数则表示将读指针(或写指针)
  • 从当前位置朝文件尾部移动 offset字节,为 0 则不移动。
  • ios::end:让文件读指针(或写指针)指向从文件结尾往前的 |offset|(offset 的绝对值)字节处。在此情况下,offset 只能是 0 或者负数。

此外,我们还可以得到当前读写指针的具体位置:

  • ifstream 类和 fstream 类还有 tellg 成员函数,能够返回文件读指针的位置;
  • ofstream 类和 fstream 类还有 tellp 成员函数,能够返回文件写指针的位置。

这两个成员函数的原型如下:

int tellg();
int tellp();

要获取文件长度,可以用 seekg 函数将文件读指针定位到文件尾部,再用 tellg 函数获取文件读指针的位置,此位置即为文件长度。

Ios::beg:表示输入流的开始位置
Ios::cur: 表示输入流的当前位置
Ios::end:表示输入流的结束位置

  • 4
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

kevin@1024

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值