C++ IO操作

参考上位:

https://blog.csdn.net/weixin_42587961/article/details/86677869
https://blog.csdn.net/weixin_40332685/article/details/81000393
https://blog.csdn.net/kingstar158/article/details/6859379
https://blog.csdn.net/vinson0526/article/details/9285003


C语言里面对文件的操作是通过文件指针,以及一些相关的函数,那么C++中是如何对文件进行操作的呢?

我们可以使用ifstream,ofstream以及fstream 这三个类了

  • ofstream是从内存到硬盘,代表一个输出文件,支持<<操作符
  • ifstream是从硬盘到内存,代表一个输入文件,支持>>操作符
  • fstream代表一个输入输出文件,支持>>和<<操作符。

open

void open ( const char * filename,  
            ios_base::openmode mode = ios_base::in | ios_base::out );  

void open(const wchar_t *_Filename,  
        ios_base::openmode mode= ios_base::in | ios_base::out,  
        int prot = ios_base::_Openprot);

  • filename:要打开的文件名

  • mode:要打开文件的方式

    ios::app:    //以追加的方式打开文件  
    ios::ate:    //文件打开后定位到文件尾,ios:app就包含有此属性  
    ios::binary:  //以二进制方式打开文件,缺省的方式是文本方式。两种方式的区别见前文  
    ios::in:     //文件以输入方式打开(文件数据输入到内存)  
    ios::out:    //文件以输出方式打开(内存数据输出到文件)  
    ios::nocreate: //不建立文件,所以文件不存在时打开失败
    ios::noreplace://不覆盖文件,所以打开文件时如果文件存在失败  
    ios::trunc:   //如果文件存在,把文件长度设为0
    

    可以用"|"把以上的值连起来。

  • prot :打开文件的属性

    0:普通文件,打开访问  
    1:只读文件  
    2:隐含文件  
    4:系统文件 
    

    可以用"|"连起来,还可以加起来,比如"3"表示以只读和隐含文件属性打开

构造函数方式打开文件

很多程序中,可能会碰到ofstream out(“Hello.txt”), ifstream in(“…”),fstream foi(“…”)这样的的使用,并没有显式的去调用open(), 函数就进行文件的操作,是因为这样是直接调用了其默认的打开方式,因为在stream类的构造函数中调用了open()函数。

默认方式

ofstream out("...", ios::out);
ifstream in("...", ios::in);
fstream foi("...", ios::in|ios::out);

ifstream默认以输入方式打开文件,如果文件不存在,则打开失败。
ofstream默认以输出方式打开文件,文件不存在,则会创建一个文件。

std::fstream i(Constants::BUNDLE_DATA_BASE_FILE);
if (!i.is_open()) {
    std::ofstream o(Constants::BUNDLE_DATA_BASE_FILE);// if file not exist, should create file here
    o.close();
    return false;
}
    std::fstream f(Constants::BUNDLE_DATA_BASE_FILE);
    bool isExist = f.good();

    if (isExist) {
		;;
    } else {
        APP_LOGI("bundle database file not exist");
        ret = false;
    }
    f.close();

指定mode方式

ifstream f("d:\\12.txt", ios::nocreate);         	//以ios::in 的方式打开文件,文件不存在时操作失败
ofstream f("d:\\12.txt");                		//以ios::out的方式打开文件
fstream  f("d:\\12.dat", ios::in|ios::out|ios::binary); 	//以读写方式打开二进制文件

状态标志符验证

所有都返回bool型返回值

is_open();	//文件是否正常打开
bad();		//读写过程中是否出错(操作对象没有打开;对一个不是打开为写状态的文件进行写入时;或者我们要写入的设备没有剩余空间的时候)
fail();		//读写过程中是否出错(除了与bad()同样的情况下会返回true以外,加上格式错误时也返回true ,例如当想要读入一个整数,而获得了一个字母的时候)
eof();		//读文件到达文件末尾,返回true
good();		//以上任何一个返回true,这个就返回false

要想重置以上成员函数所检查的状态标志,你可以使用成员函数clear(),没有参数

检查是否成功打开

成功

if (f) {...}       //对ifstream、ofstream对象可 用,fstream对象不可用。 mysql
if (f.good()) {...}

失败

if (!f) {...}       // !运算符已经重载
if (f.fail()) {...}

关闭文件

当文件读写操作完成之后,我们必须将文件关闭以使文件重新变为可访问的。成员函数close(),它负责将缓存中的数据排放出来并关闭文件。

这个函数一旦被调用,原先的流对象就可以被用来打开其它的文件了,这个文件也就可以重新被其它的进程所访问了

为防止流对象被销毁时还联系着打开的文件,析构函数将会自动调用关闭函数close。

读写文件

读写文件分为文本文件二进制文件的读取,

文本文件

对于文本文件的读取比较简单,用插入器<<和析取器>>就可以了,也可以使用getline等函数;

dec 格式化为十进制数据
hex 格式化为十六进数据
oct 格式化为八进制数据

endl 输出一个换行符并刷新此流输出 
ends 输出一个空字符输出 

setpxecision(int p) 设置浮点数的精度位数

比如要把123当作十六进制输出:

file1<<hex<<123<<endl;
//or
file<<setbase(16)<<123<<endl;

示例

// writing on a text file
#include <ofstream.h>

int main () {
    ofstream out("out.txt");
    if (out.is_open()) 
    {
        out << "This is a line.\n";
        out << "This is another line.\n";
        out.close();
    }
    return 0;
}
//结果: 在out.txt中写入:
//This is a line.
//This is another line 
 // reading a text file
 #include <iostream.h>
 #include <fstream.h>
 #include <stdlib.h>
    
int main ()
{
    char buffer[256];
    ifstream in("test.txt");
    if (! in.is_open()){
        cout << "Error opening file"; exit (1);
    }
    
    while (!in.eof() ){
        in.getline (buffer,100);
        cout << buffer << endl;
    }
    
    return 0;
}
//结果 在屏幕上输出
//This is a line.
//This is another line
#include<fstream>
 
void main()
{
    fstream f("d:\\try.txt", ios::out);
    f << 1234 << ' ' << 3.14 << 'A' << "How are you"; //写入数据
    f.close();
    f.open("d:\\try.txt", ios::in);
    
    int i;
    double d;
    char c;
    char s[20];
    f >> i >> d >> c;               //读取数据
    f.getline(s,20);
    cout << i << endl;             //显示各数据
    cout <<d << endl;
    cout << c << endl;
    cout << s << endl;
    f.close();
}

运行结果:

1234
3.14
A
How are you
Press any key to continue

二进制文件

在二进制文件中,使用<< 和>>,以及函数(如getline)来操作符输入和输出数据,没有什么实际意义,虽然它们是符合语法的。

put();

put()函数向流写入一个字符,其原型

ofstream &put(char ch);

​ 如: file1.put(‘c’);

get();

get()函数比较灵活,有3种常用的重载形式:

ifstream &get(char&ch);//功能是从流中读取一个字符,结果保存在引用ch中,如果到文件尾,返回空字符。

​ 如: file2.get(x);

int get();//这种形式是从流中返回一个字符,如果到达文件尾,返回EOF

​ 如: x=file2.get();

ifstream &get(char *buf,int num,char delim='\n');//把字符读入由 buf 指向的数组,直到读入了 num 个字符或遇到了由 delim 指定的字符,如果没使用 delim 这个参数,将使用缺省值换行符'\n'。

读写数据块

read(unsigned char *buf,int num);
write(const unsigned char *buf,int num);
  • read()从文件中读取 num 个字符到 buf 指向的缓存中,如果在还未读入 num 个字符时就到了文件尾,可以用成员函数 int gcount();来取得实际读取的字符数;

  • read 是istream 的一个成员函数,被ifstream 所继承。

  • write() 从buf 指向的缓存写 num 个字符到文件中,值得注意的是缓存的类型是 unsigned char *,有时可能需要类型转换。

  • write是ostream 的一个成员函数,被ofstream所继承。

读写二进制文件注意事项
打开方式中必须指定ios::binary,否则读写会出错

用read\write进行读写操作,不能使用插入、提取运算符进行操作,否则会出错。

示例:

// print the content of a text file.
#include <iostream>     // std::cout
#include <fstream>      // std::ifstream

int main () {
  std::ifstream ifs;

  ifs.open ("test.txt", std::ifstream::in);

  char c = ifs.get();

  while (ifs.good()) {
    std::cout << c;
    c = ifs.get();
  }

  ifs.close();

  return 0;
}

//使用get()一次读一个字符--------------------------------方案一
#include<fstream>
 
void main()
{
    ifstream fin("d:\\简介.txt", ios::nocreate);
    if (!fin) {
        cout << "File open error!\n";
        return;
	}
    
    char c;
    while ((c=fin.get()) != EOF)
        cout << c;    //注意结束条件的判断
    fin.close();
}

巧妙利用文本文件中不会有字符’\0’的特点进行读取

#include<fstream>

//使用get(char *,int n,char delim='\n')一次读多个字符
void main()
{
    ifstream fin("d:\\简介.txt",ios::nocreate);
    if(!fin){
        cout<<"File open error!\n";
        return;
	}
    
    char c[80];
    while(fin.get(c,80,'\0')!=NULL)
        cout<<c; //注意结束条件的判断
    fin.close();
}
//使用read(char *,int n)读文件
#include<fstream.h>
void main()
{
    ifstream fin("d:\\简介.txt",ios::nocreate);
    if(!fin){
        cout<<"File open error!\n";
        return;
    }
    
    char c[80];
    while(!fin.eof()){            //判断文件是否读结束
        fin.read(c,80);
        cout.write(c,fin.gcount());
    }
    
    fin.close();
}

拷贝文件

#include<fstream>
 
void main()
{
    ifstream fin("C:\\1.exe", ios::nocreate|ios::binary);
    if (!fin) {
        cout << "File open error!\n";
        return;
    }
    
    ofstream fout("C:\\2.exe", ios::binary);
    char c[1024];
    while (!fin.eof()){
        fin.read(c, 1024);
        fout.write(c, fin.gcount());
    }
    
    fin.close();
    fout.close();
    cout << "Copy over!\n";
}

文件定位

和C的文件操作方式不同的是,C++ I/O系统管理两个与文件相关联的指针。

  • 一个是读指针,它说明输入操作在文件中的位置;
  • 一个是写指针,它下次写操作的位置。

每次执行输入或输出时,相应的指针自动变化。所以,C++的文件定位分为读位置和写位置的定位,

file1.seekg(1234,ios::cur);//把文件的读指针从当前位置向后移1234个字节
file2.seekp(1234,ios::beg);//把文件的写指针从文件开头向后移1234个字节

函数汇总

seekg(绝对位置);      //绝对移动,    //输入流操作
seekg(相对位置,参照位置);  //相对操作
tellg();          //返回当前指针位置

seekp(绝对位置);      //绝对移动,    //输出流操作
seekp(相对位置,参照位置);  //相对操作   
tellp();          //返回当前指针位置

参照位置

ios::beg  = 0       //相对于文件头
ios::cur   = 1      //相对于当前位置
ios::end  = 2       //相对于文件尾

tellg() 和 tellp()

这两个成员函数不用传入参数,返回pos_type 类型的值(根据ANSI-C++ 标准) ,就是一个整数,代表当前get 流指针的位置 (用tellg) 或 put 流指针的位置(用tellp).

seekg() 和seekp()

这对函数分别用来改变流指针get 和put的位置。两个函数都被重载为两种不同的原型:

seekg ( pos_type position );
seekp ( pos_type position );

使用这个原型,流指针被改变为从文件开始计算的一个绝对位置。要求传入的参数类型与函数 tellg 和tellp 的返回值类型相同。

seekg ( off_type offset, seekdir direction );
seekp ( off_type offset, seekdir direction );

使用这个原型可以指定由参数direction决定的一个具体的指针开始计算的一个位移(offset)。它可以是ios::beg、ios::cur、ios::end。

这两个函数一般用于二进制文件,因为文本文件会因为系统对字符的解释可能与预想的值不同。

由于这个原因,建议对以文本文件模式打开的文件总是使用seekg 和 seekp的第一种原型,而且不要对tellg 或 tellp 的返回值进行修改。对二进制文件,你可以任意使用这些函数,应该不会有任何意外的行为产生。

获得一个二进制文件的大小

 // obtaining file size
    #include <iostream.h>
    #include <fstream.h>
    
    const char * filename = "test.txt";
    
    int main () {
        long l,m;
        ifstream in(filename, ios::in|ios::binary);
        l = in.tellg();
        in.seekg (0, ios::end);
        m = in.tellg();
        in.close();
        cout << "size of " << filename;
        cout << " is " << (m-l) << " bytes.\n";
        return 0;
    }
   
   //结果:
   size of example.txt is 40 bytes.
#include <iostream>
#include <fstream.h>
    
const char * filename = "test.txt";
    
int main () 
{
    char * buffer;
    long size;
    ifstream in (filename, ios::in|ios::binary|ios::ate);
   
    size = in.tellg();
    in.seekg (0, ios::beg);
    buffer = new char [size];
    
    in.read (buffer, size);
    in.close();

    cout << "the complete file is in a buffer";

    delete[] buffer;
    return 0;
}

缓存和同步

当我们对文件流进行操作的时候,它们与一个streambuf 类型的缓存(buffer)联系在一起。这个缓存(buffer)实际是一块内存空间,作为流(stream)和物理文件的媒介。例如,对于一个输出流, 每次成员函数put (写一个单个字符)被调用,这个字符不是直接被写入该输出流所对应的物理文件中的,而是首先被插入到该流的缓存(buffer)中。

当缓存被排放出来(flush)时,它里面的所有数据才被写入物理媒质中,这个过程称为同步(synchronization),它会在以下任一情况下发生:

  • 当文件被关闭时: 在文件被关闭之前,所有还没有被完全写出或读取的缓存都将被同步。
  • 当缓存buffer满时:缓存Buffers 有一定的空间限制。当缓存满时,它会被自动同步。
  • 控制符明确指明:当遇到流中某些特定的控制符时,同步会发生。这些控制符包括:flush 和endl。
  • 明确调用函数sync(): 调用成员函数sync() (无参数)可以引发立即同步。这个函数返回一个int 值,等于-1 表示流没有联系的缓存或操作失败。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
 一、ASCII 输出   为了使用下面的方法, 你必须包含头文件<fstream.h>(译者注:在标准C++中,已经使用<fstream>取 代< fstream.h>,所有的C++标准头文件都是无后缀的。)。这是 <iostream.h>的一个扩展集, 提供有缓 冲的文件输入输出操作. 事实上, <iostream.h> 已经被<fstream.h>包含了, 所以你不必包含所有这两个 文件, 如果你想显式包含他们,那随便你。我们从文件操作类的设计开始, 我会讲解如何进行ASCII I/O 操作。如果你猜是"fstream," 恭喜你答对了! 但这篇文章介绍的方法,我们分别使用"ifstream"?和 "ofstream" 来作输入输出。   如果你用过标准控制台流"cin"?和 "cout," 那现在的事情对你来说很简单。 我们现在开始讲输出部 分,首先声明一个类对象。 ofstream fout;   这就可以了,不过你要打开一个文件的话, 必须像这样调用ofstream::open()。 fout.open("output.txt");   你也可以把文件名作为构造参数来打开一个文件. ofstream fout("output.txt");   这是我们使用的方法, 因为这样创建和打开一个文件看起来更简单. 顺便说一句, 如果你要打开的文 件不存在,它会为你创建一个, 所以不用担心文件创建的问题. 现在就输出到文件,看起来和"cout"的操 作很像。 对不了解控制台输出"cout"的人, 这里有个例子。 int num = 150; char name[] = "John Doe"; fout << "Here is a number: " << num << " "; fout << "Now here is a string: " << name << " ";   现在保存文件,你必须关闭文件,或者回写文件缓冲. 文件关闭之后就不能再操作了, 所以只有在你 不再操作这个文件的时候才调用它,它会自动保存文件。 回写缓冲区会在保持文件打开的情况下保存文 件, 所以只要有必要就使用它。回写看起来像另一次输出, 然后调用方法关闭。像这样: fout << flush; fout.close();    现在你用文本编辑器打开文件,内容看起来是这样:   Here is a number: 150 Now here is a string: John Doe   很简单吧! 现在继续文件输入, 需要一点技巧, 所以先确认你已经明白了流操作,对 "<<" 和">>" 比较熟悉了, 因为你接下来还要用到他们。继续…   二、ASCII 输入   输入和"cin" 流很像. 和刚刚讨论的输出流很像, 但你要考虑几件事情。在我们开始复杂的内容之前 , 先看一个文本:   12 GameDev 15.45 L This is really awesome!   为了打开这个文件,你必须创建一个in-stream对象,?像这样。 ifstream fin("input.txt");   现在读入前四行. 你还记得怎么用"<<" 操作符往流里插入变量和符号吧?好,?在 "<<" (插入)?操作 符之后,是">>" (提取) 操作符. 使用方法是一样的. 看这个代码片段.
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值