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”转换后得到错误的路径,因此文件打不开!