C++读写文件

C++文件流的头文件为,对应的类有3个:
std::fstream // 文件流,由iostream派生而来,用于读写文件
std::ifstream // 输入文件流,由istream派生而来, 用于读文件
std::ofstream // 输出文件流,由ostream派生而来,用于写文件

1.文件操作

1)打开文件

  • 文件名
    注意路径名中的斜杠要双写,如:
    “D:\MyFiles\ReadMe.txt”

  • 文件打开方式选项
    ios::in    = 0x01, //供读,文件不存在则创建(ifstream默认的打开方式)
    ios::out    = 0x02, //供写,文件不存在则创建,若文件已存在则清空原内容(ofstream默认的打开方式)
    ios::ate    = 0x04, //文件打开时,指针在文件最后。可改变指针的位置,常和in、out联合使用
    ios::app    = 0x08, //供写,文件不存在则创建,若文件已存在则在原文件内容后写入新的内容,指针位置总在最后
    ios::trunc   = 0x10, //在读写前先将文件长度截断为0(默认)
    ios::nocreate = 0x20, //(已不再支持)文件不存在时产生错误,常和in或app联合使用
    ios::noreplace = 0x40, //(已不再支持)文件存在时产生错误,常和out联合使用
    ios::binary  = 0x80  //二进制格式文件,若文件已存在则清空原内容

  • 文件保护方式选择项
    filebuf::openprot;   //默认的兼容共享方式
    filebuf::sh_none;    //独占,不共享
    filebuf::sh_read;    //读共享
    filebuf::sh_write;   //写共享

  • 打开文件的方法

    • 调用构造函数时指定文件名和打开模式
      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); //以读写方式打开二进制文件
    • 使用Open成员函数
      fstream f;
      f.open(“d:\12.txt”,ios::out); //利用同一对象对多个文件进行操作时要用到open函数

2)检查是否成功打开

  • 成功
    if(f){…} //对ifstream、ofstream对象可用,fstream对象不可用。
    if(f.good()){…}
  • 失败
    if(!f){…}       // !运算符已经重载
    if(f.fail()){…}

3)读写操作

  • 使用<<,>>运算符
    只能进行文本文件的读写操作,用于二进制文件可能会产生错误。
  • 使用函数成员 get、put、read、write等
    经常和read配合使用的函数是gcount(),用来获得实际读取的字节数。
  • 读写二进制文件注意事项
    打开方式中必须指定ios::binary,否则读写会出错
    用read\write进行读写操作,而不能使用插入、提取运算符进行操作,否则会出错。
    使用eof()函数检测文件是否读结束,使用gcount()获得实际读取的字节数

4)关闭文件

  • 使用成员函数close,如:
    f.close(); 
  • 利用析构函数
    对象生命期结束时会检查文件是否关闭,对没有关闭的文件进行关闭操作。

5)随机读写文件

  • 通过移动文件读写指针,可在文件指定位置进行读写。
    seekg(绝对位置); //绝对移动,    //输入流操作
    seekg(相对位置,参照位置); //相对操作
    tellg(); //返回当前指针位置
    seekp(绝对位置); //绝对移动,    //输出流操作
    seekp(相对位置,参照位置); //相对操作
    tellp(); //返回当前指针位置
  • 参照位置
    ios::beg = 0 //相对于文件头
    ios::cur = 1 //相对于当前位置
    ios::end = 2 //相对于文件尾

2. 代码示例

1)读取并在窗口显示文件内容

方案一:使用get()一次读一个字符

#include <iostream>
#include <fstream>

int main()
{
    std::ifstream fin("d:\\a.txt");
    if(!fin)
    {
        std::cout<<"File open error!\n";
        return 1;
    }
    char c;
    while((c=fin.get())!=EOF)   //注意结束条件的判断
    {
        std::cout<<c;
    }
    fin.close();
    return 0;
}

方案二:使用get(char *,int n,char delim=’\n’)一次读多个字符

#include <iostream>
#include <fstream>

void main()
{
    std::ifstream fin("d:\\a.txt");
    if(!fin)
    {
        std::cout<<"File open error!\n";
        return;
    }
    char c[80];
    while(fin.get(c,80,'\0')!=NULL) //注意结束条件的判断
    {
        std::cout<<c;
    }
    fin.close();
}

方案三:使用read(char *,int n)读文件

#include <iostream>
#include <fstream>

void main()
{
    std::ifstream fin("d:\\a.txt");
    if(!fin)
    {
        std::cout<<"File open error!\n";
        return;
    }
    char c[80];
    while(!fin.eof())   //判断文件是否读结束
    {
        fin.read(c,80);
        std::cout.write(c,fin.gcount()); 
    }
    fin.close();
}

2)拷贝文件,二进制文件操作示例

#include <iostream>
#include <fstream>

void main()
{
    std::ifstream fin("d:\\a.data", std::ios::binary);
    if(!fin)
    {
        std::cout<<"File open error!\n";
        return;
    }
    std::ofstream fout("d:\\a2.data", std::ios::binary);
    char c[1024];
    while(!fin.eof())
    {
        fin.read(c,1024);
        fout.write(c,fin.gcount());
    }
    fin.close();
    fout.close();
    std::cout<<"Copy over!\n";
}

注意,如果文件路径或者文件名称包含中文,需要特殊处理,如下:

#include <locale>

std::locale::global(std::locale(""));   //将全局区域设为操作系统默认区域
std::ifstream fin("d:\\测试.txt");
std::locale::global(std::locale("C"));  //还原全局区域设定

原因为:

在调用ifstream的open方法时,系统内部调用mbstowcs_s进行文件名转换(mbstowcs_s函数的作用是把多字节字符转化为宽字符),需要注意的是,该函数的调用结果依赖于程序的本地化设置(什么是本地化设置?)。而本地化设置可以通过setlocale函数来设置,譬如:setlocale(LC_ALL, “chinese”)表示将程序本身的语言设置为中文,而程序启动时默认设置为LC_ALL=”C”。在使用mbstowcs_s进行字符串转换时,只有当LC_ALL=”chinese”时,含中文的字符串才能正确的转换成其对应的宽字节字符,否则(在LC_ALL=”C”时),汉字会被看成2个单字节的字符,然后再转换成宽字节的字符,这样转换的结果显然是错误的!这就是ifstream打开含中文路径的文件失败的原因,因为”d:\测试.txt”转换后得到错误的路径,因此文件打不开!

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值