C++的I/O流体系
继承关系:
ios <-- istream/ostream
|--> istream <-- istrstream/ifstream/iostream
|--> ostream <-- ostrstream/ofstream/iostream
|--> iostream <-- strstream/fstream
istream及其子类支持输入操作
ostream及其子类支持输出操作
由istream和ostream共同派生的子类iostream既支持输入也支持输出。
1.通过ofstream类单参(文件路径)构造函数可以打开一个文件用于向文件中输出,如果文件不存在,该类的构造函数将以缺省的方式创建文件;
如果文件存在,进行覆盖
2.ofstream的两参数(文件路径,打开模式openmode)版本
ios::in:打开文件用于从文件中读取
ios::out:打开文件用于向文件中写入
ios::app:打开文件用于向文件中追加
ios::ate:打开文件,把文件位置设置到文件尾部,用于从文件读取
// 在打开文件流时使用ios::ate选项,将文件的get位置置于文件尾
ifstream ifs ("文件路径", ios::binary | ios::ate);
if (! ifs)
PrintError (__FILE__, __LINE__, "Cannot open input file");
size_t uBytes = ifs.tellg (); // 通过tellg()函数获取当前的get位置, 以此计算出文件的总字节数
char* pcBuffer = new char[uBytes]; // 有了总字节数,我们就可以分配足 够的内存,一次性地读取文件的全部内容
ifs.seekg (ios::beg); // 在读取之前一定要将文件的 get位置重置于文件起始处
ifs.read (pcBuffer, uBytes);//读取文件内容
ios::trunc: 打开文件的同时删除其原有内容,用于向文件中输入
ios::binary:以二进制模式打开文件,用于输入/输出
ofstream ofs ("xxx", ios::out|ios::trunc|ios::binary);
if (! ofs)
3.ifstream和ofstream用法基本相同,用于从文件中读取
但是ifstream要求指定的文件必须存在
4.输入流的提取操作>>在格式化提取的过程中,以空格、制表符、换行分隔
5.fstream以读和写(默认)的方式打开一个文件时,文件必须存在
6.tellg/tellp:获取当前文件位置(包括\n等)的下一个
p表示put、g表示get
tellp在将get位置调整到文件开始的同时,也改变了put位置
seekg/seekp:设置当前文件的get/set位置
1.单参: seekg(0)与seekg(ios::beg)完成相同的任务
2.双参:
第一个参数代表相对于第二个参数所指定位置的偏移量:
ios::beg:文件头如:fs.seekg (5, ios::beg);//从文件头(开始下标为0)开始数5个字符
如:1234 56.78 apples;//即5数字开始操作
ios::end:文件尾
ios::cur:当前位置如:fs.seekg (-6, ios::cur);
在上面的情况下:tellp/tellg都获取值为:0
如果文件位置被移到了文件的起始位置之前或结束位置之后,那么该文件流将进入一种错误状态。
这时程序对这个流所做的更进一步处理将会被暂停,直到我们通过clear() 函数清除了这种错误状态为止。
如:
#include
#include
using namespace std;
int main (int argc, char* argv[]) {
ofstream ofs1 ("ex01.txt");
if (! ofs1)
cerr << "Cannot open output file" << endl;
ofs1 << 1234 << " " << 56.78 << " " << "apples" << '\n';
ofs1.close ();
ofstream ofs2 ("ex01.txt", ios::app);
if (! ofs2)
cerr << "Cannot open output file" << endl;
ofs2 << "append_a_line\n";
ofs2.close ();
ifstream ifs1 ("ex01.txt");
if (! ifs1)
cerr << "Cannot open input file" << endl;
int n;
double f;
string str1, str2;
ifs1 >> n >> f >> str1 >> str2;
ifs1.close ();
cout << n << " " << f << " " << str1 << " " << str2 << endl;
////////////////////////////////////////////////////////////
fstream fs ("ex02.txt", ios::in | ios::out);
if (! fs)
cerr << "Cannot open file" << endl;
fs << 1234 << " " << 56.78 << " " << "apples" << '\n';
cout << fs.tellg () << endl;//获取当前文件位置
cout << fs.tellp () << endl;//获取当前文件位置
fs.seekg (5, ios::beg);//同seekp,都是设置当前文件位置
fs << 1;
fs.seekg (-6, ios::cur);
cout << fs.tellg () << endl;
cout << fs.tellp () << endl;
int n;
double f;
string str;
fs >> n >> f >> str;
fs.close ();
cout << n << " " << f << " " << str << endl;
return 0;
}
7.io的非格式化:
(1).ofs.put(char);//一次写入一个字符
(char = ifs.get()) != EOF//一次获取一个字符,不为文件结尾
(2).get()的三参数:定界符并不被提取,但文件位置会停在定界符处,下次读取会发生错误
char sz[4];
while (ifs.get (sz, sizeof (sz), '\n'))//自定义'\n'为定界符
{
cout << sz;
cout.flush ();//刷新输入输出才可能显示出来(字符少的时候)
//注意大小为4的sz可能没有被填满,因为定界符不被get读取,所以要跳过
if (ifs.peek () == '\n')//读当前位置的字符,但不改变文件位置
ifs.ignore ();//掠过当前位置的字符,文件位置后移
}
if (! ifs.eof ())//当前是否到了文件结尾
cerr << "Cannot read input file 1" << endl;
ifs.clear ();
(3).getline():可以自动掠过定界符,但当其试图读取第二个参数-1个字符时会发生错误
因为当文件结束的时候,可能没有填满第二个参数-1个字符,这时又结束了(读取不到字符),那么就会出错
char szLarge[8];
while (ifs.getline (szLarge, sizeof (szLarge), '\n'))
cout << szLarge;
if (! ifs.eof ())//当前是否到了文件结尾
cerr << "Cannot read input file" << endl;
8.字节流:
read(char*, size_t)/write(char*, size_t)
如:
char caBuffer[256];
while (ifs.read (cbBuffer, sizeof (cbBuffer)){
ofs.write (cbBuffer, sizeof (cbBuffer));
}
//可通过gcount()函数推断出实际读到byBuffer缓冲区中的字节数。
if (ifs.eof ())//进入某种错误状态时亦返回false
ofs.write (cbBuffer, ifs.gcount ());
else
cerr << "error!" << endl;
9.流函数:
#include
#include
#include
#include
using namespace std;
int main (int argc, char* argv[]) {
cout << sqrt (2) << endl;//1.41421
cout.precision (10);//设置小数点精度,区别不带参数的函数(返回精度)
cout << sqrt (2) << endl;//1.414213562
cout << setprecision (10) << sqrt (2) << endl;//1.414213562
cout << sqrt (2) << endl;//1.414213562
cout << cout.precision () << endl;//10
cout << setprecision (2) << 1.24 << ", " << 1.25 << ", " << 1.26 << endl;//1.2, 1.3, 1.3
cout << oct << 127 << endl;//177
cout << hex << 127 << endl;//7f
cout << dec << 127 << endl;//127
cout << setw (7)/*注意这个函数不保存到下面的设置*/ << oct << 127 << endl;//177
cout << setw (7) << hex << 127 << endl;//7f
cout << setw (7) << dec << 127 << endl;//127
//流控制符
cout << setfill ('#') << setw (7) << oct << 127 << endl;//####177
cout << setfill ('@') << setw (7) << hex << 127 << endl;//@@@@@7f
cout << setfill ('%') << setw (7) << dec << 127 << endl;//%%%%127
cout << left << setfill ('#') << setw (7) << oct << 127 << endl;//177####
cout << left << setfill ('@') << setw (7) << hex << 127 << endl;//7f@@@@@
cout << left << setfill ('%') << setw (7) << dec << 127 << endl;//127%%%%
cout << showbase << oct << 127 << endl;//0177
cout << hex << 127 << endl;//0x7f
cout << dec << 127 << endl;//127
cout << noshowbase << oct << 127 << endl;//177
cout << dec << 127 << endl;//127
cout << 12.00 << endl;//12
cout << setprecision (10) << showpoint << 12.00 << endl;//12.00000000
cout.setf (ios::showpoint);//显示后面的0
cout << 12.00 << endl;//12.00000000
cout.precision (4);
cout << sqrt (200) << endl;//14.14
cout.setf (ios::scientific);/*精度所限制的实际上只是底数小数部分的显示位数*/
cout << sqrt (200) << endl;//1.4142e+001
ifstream ifs ("ex04.txt");
ifs.unsetf (ios::skipws);/*跳过空白字符(空格符、制表符以及换行 符等),
但如果出于某种原因,我们不希望忽略输入流中的空白字符,
那么我们就可以使用控制符noskipws或者调用输入流的unsetf(ios::skipws)成员函数。
我们也可以使用操纵子skipws来取消noskipws的效果,或者调
用输入流的setf(ios::skipws)成员函数达到同样的目的*/
char c;
while (ifs >> c)
cout << c;
return 0;
}
10.字符串流:
(1)ostringstream对象的一个优点是,可以利用输出操作符“<<”的格式化能力把数值类型的数据转换为它们的字符串表示形式
(2)另外:格式化字符数组
char s[1024];
sprintf (s, "money = %.2f\n", 23.44);