19.1、文本IO
如果你需要写入一千份以上的数据,如果手打那可太浪费时间了。这种情况下,应该让程序直接读取文件;同样的,让程序将输入写入到文件也是更加方便。
文本I/O的概念:使用cin进行输入时,程序将输入视为一系列的字节,其中每个字节都被解释为字符编码。无论目标数据类型是什么,输入一开始都是文本数据,然后cin对象再将这些文本转换为其他类型。对于输出,将执行相反的转换。即整数被转换为数字字符序列,浮点数被转换为数字字符和其他字符组成的字符序列。
19.2、写入到文本文件中
我们这里复习一下cout用于控制台输出的东西:
(1)必须包含头文件iostream
(2)头文件iostream定义了用处理输出的ostream类
(3)头文件iostream声明了一个名为cout的ostream变量(对象)
(4)必须指明名称空间std;为了引用cout和endl,必须使用编译指令using或者前缀std::
(5)可以结合使用cin和运算符>>来读取各种类型的数据
文件输出与此极其相似:
(1)必须包含头文件fstream
(2)头文件fstream定义了一个用于处理输出的ofstream类
(3)需要声明一个或多个ofstream变量(对象),并以自己喜欢的方式对其进行命名,条件是遵守常用的命名规则
(4)必须指明名称空间std;为了引用元素ofstream,必须使用编译指令using或前缀std::
(5)需要将ofstream对象与文件关联起来,方法之一是open()方法
(6)使用完文件后,应使用close()方法将其关闭
(7)可结合使用ofstream对象和运算符<<来读取各种类型的数据
虽然头文件iostream提供了一个预先预定好的名为cout的ostream对象,但你必须声明自己的ofstream对象,为其命名,并将其同文件关联起来。
下面演示一下如何声明这些对象:
ofstream outFile;
ofstream fout;
下面演示如何将这种对象与特定的文件关联起来:
outFile.open("fish.txt");
char filename[50];
cin>>filename;
fout.open(filename);
方法open()结束一个C风格字符串作为参数,这可以是一个字面字符串,也可以是存储在数组中的字符串。
double wt=125.8;
outFile<<wt;
char line[80]="Objects are closer than they appear.";
fout<<line<<endl;
重要的是,声明一个ofstream对象并将其同文件关联起来后,便可以想使用cout那样使用它。所有可用于cout的操作方法都可以用于ofstream对象。
总之使用文件输出的主要步骤如下:
1、包含头文化fstream
2、创建一个ofstream对象
3、将爱ofstream对象同一个文件关联起来
4、像使用cout那样使用该ofstream对象。
下面这段程序演示了这种方法,它要求用户输入信息,然后将信息显示到屏幕上,再将这些信息写入到文件中:
#include <iostream>
#include <fstream>
int main()
{
using namespace std;
char automobile[50];
int year;
double a_price;
double d_price;
ofstream outFile;
outFile.open("carinfo.txt");
cout << "Enter the make and model of automobile: ";
cin.getline(automobile, 50);
cout << "Enter the model year: ";
cin >> year;
cout << "Enter the original asking price: ";
cin >> a_price;
d_price = 0.913 * a_price;
cout << fixed;
cout.precision(2);
cout.setf(ios_base::showpoint);
cout << "Make and model: " << automobile << endl;
cout << "Year: " << year << endl;
cout << "Was asking $" << a_price << endl;
cout << "Now asking $" << d_price << endl;
outFile << fixed;
outFile.precision(2);
outFile.setf(ios_base::showpoint);
outFile << "Make and model: " << automobile << endl;
outFile << "Year: " << year << endl;
outFile << "Was asking $" << a_price << endl;
outFile << "Now asking $" << d_price << endl;
outFile.close();
return 0;
}
屏幕输出是使用cout的结果,如果你查看该程序的可执行文件所在的目录,将看到一个名为carinfo.txt的新文本(也可能在其他文件夹里),查看后你会发现这些:
在程序里,声明一个ofstream对象后,便可以使用open()将该对象特定文件关联起来:
ofstream outFile;
outFile.open("carinfo.txt");
程序使用完文件后,应该将其关闭:
outFile.close();
注:close()方法不需要使用文件名作为参数,这是因为outFile已经同特定的文件关联起来。如果你忘记关闭文件,程序正常运行也会自动关闭它。
回到open()方法:
outFile.open("carinfo.txt");
在程序运行之前,文件carinfo.txt其实并不存在。在这种情况下,方法open()将新建一个名为carinfo.txt的文件。如果再次运行该程序将会发生什么?默认情况下,程序将截断该文件(即丢弃原有的文件),然后将的输出加到该文件中。打开文件可能会失败,例如指定文件已存在但禁止访问,后面会讲如何解决。
19.3、读取文本文件
文件输入其实与输出差不多:
(1)必须包含头文件fstream
(2)头文件fstream定义了一个用于处理输入的ifstream类
(3)需要声明一个或多个ifstream变量(对象),并以自己喜欢的方式对其进行命名,条件是遵守常用的命名规则
(4)必须指明名称空间std;为了引用元素ifstream,必须使用编译指令using或前缀std::
(5)需要将ifstream对象与文件关联起来,方法之一是open()方法
(6)使用完文件后,应使用close()方法将其关闭
(7)可结合使用ifstream对象和运算符>>来读取各种类型的数据
(8)可以使用ifstream对象和get()方法来读取一个字符,使用ifstream对象和getline()来读取一行字符
(9)可以结合使用ifstream和eof()、fail()等方法来判断输入是否成功
(10)ifstream对象本身被用作测试条件时,如果最后一个读取成功,它将被转换为布尔值true,否则被转换为false
与写入文本文件一样,也要声明自己的ifstream对象,为其命名,并将其同文件关联起来:
ifstream inFile;
ifstream fin;
inFile.open("bowling.txt");
char filename[50];
cin>>filename;
fin.open(filename);
如果试图打开一个不存在的文件用于输入,将导致后面使用ifstream对象进行输入时失败。检查文件是否被成功打开的首先方法是使用方法is_open(),因此我们可以使用以下代码:
inFile.open("bowling.txt");
if(!inFile.is_open())
{
exit(EXIT_FAILURE);
}
如果文件被成功地打开,方法is_open()将返回true;如果文件没有被打开,表达式!inFile.is_open()将为true。函数exit()的原型是在头文件cstdlib中定义的,在该头文件中,还定义了一个用于同操作系统通信的参数值EXIT_FAILURE。函数exit()终止程序。
#include <iostream>
#include <fstream>
#include <cstdlib>
const int SIZE = 60;
int main()
{
using namespace std;
char filename[SIZE];
ifstream inFile;
cout << "Enter name of data file: ";
cin.getline(filename,SIZE);
inFile.open(filename);
if (!inFile.is_open())
{
cout << "Could not open the file " << filename << endl;
cout << "Program terminating." << endl;
exit(EXIT_FAILURE);
}
double value;
double sum = 0.0;
int count = 0;
inFile >> value;
while (inFile.good())
{
++count;
sum += value;
inFile >> value;
}
if (inFile.eof())
cout << "End of file reached." << endl;
else if (inFile.fail())
cout << "Input terminated by data mismatch." << endl;
else
cout << "Input terminated for unknown reason." << endl;
if (count == 0)
cout << "No data processd." << endl;
else
{
cout << "Items read: " << count << endl;
cout << "Sum: " << sum << endl;
cout << "Average: " << sum / count << endl;
}
inFile.close();
return 0;
}
要运行这个程序,首先必须创建一个包含数字的文本文件,我在电脑上新建一个文件名为Furina.txt,并包含以下数据:(当然最好放在同程序同一个文件夹里)
注:Windows文本文件的每行都以回车字符或者换行符结尾,有些文本编辑器不会自动在最后一行末尾加上换行符,因此最好在文本末尾按下回车键再保存。
该程序没有使用硬编码文件名,而是将用户提供的文件名存储到字符数组filename中,然后作为open()的参数。
正如前面所说,检查文件是否打开非常重要。读取文件有几点需要检查:首先,程序读取文件时不应超过EOF;其次程序可能遇到类型不匹配的情况,最后是文件受损或者硬件问题。