1、第10章 输入输出流 课程要求1数据输入输出的基本方法2.文件操作3.字符串流的操作主要内容10.1 C+的输入输出10.2 标准输入输出流10.3 文件操作与文件流10.4 字符串流10.1 C+的输入输出 C+的输入输出是以流(字节序列)的形式进行的。C+的输入输出主要有以下三种:(1)对系统指定的标准设备的输入输出,即标准标准I/O。(2)以外存磁盘文件为对象进行的输入输出,称为文件的输入输出文件的输入输出。(3)对内存指定的空间进行输入输出,称为字符串字符串输入输出输入输出。10.1.1 C+流的概念 1流与流类C+的输入输出流是由若干字节组成的字节序列,这些字节中的数据按一定的顺序从
2、一个对象传送到另一个对象。从流中获取数据的操作称为提取操作,向流中添加数据的操作称为插入操作。C+提供了一些供程序设计者使用的类,在这些类中封装了可以实现输入输出操作的函数,这些类统称为I/O流类。流是用流类定义的对象,如cin、cout 2流的分类(1)文本流和二进制流:按照数据的格式(2)缓冲流与非缓冲流:按是否使用缓冲区10.1.2 C+流类库 C+的流类库是用继承方法建立起来的输入输出类库,由支持标准输入输出操作的基类和支持特定种类的源和目标的输入输出操作的类组成。它具有两个平行的基类,所有其它的流类都是从它们直接或间接地派生出来的。streambuf类:提供对流缓冲区的低级操作 io
3、s类:提供对设备、文件的读写操作 1通用I/O流类库(1)基类ios(2)输入类istream(3)输出类ostream(4)输入输出类iostream2文件I/O流类库(1)ofstream类(2)ifstream类(3)fstream 类3字符串I/O流类库(1)ostrstream 类(2)istrstream 类(3)strstream类10.1.3 与iostream类库有关的头文件(1)iostream.h:包含对输入输出流进行操作的基本信息,提供无格式支持的低级输入输出和有格式支持的高级输入输出操作功能。(2)fstream.h:包含管理文件输入输出操作的有关信息。(3)strs
4、trea.h:包含对内存中数据进行输入输出操作的有关信息。(4)stdiostream.h:包含进行C风格的输入输出操作的有关信息。(5)iomanip.h:包含输入输出流的格式控制符(manipulator)的有关信息。当一个程序中需要进行标准流输入输出操作时,则必须包含头文件iostream.h;当需要进行文件流输入输出操作时,则必须包含头文件fstream.h;当需要进行字符串流读/写操作时,则必须包含头文件strstream.h。10.1.4 插入与提取运算符的重载“”本来在C+中被定义为左位移运算符和右位移运算符,由于在文件iostream.h中对它们进行了重载,使得它们能用作标准类
5、型数据的输入输出运算符。从流中获取数据的操作称为提取操作,使用提取运算符“”;向流中添加数据的操作称为插入操作,使用插入运算符“”。10.2 格式化输入输出格式化输入输出通过键盘和屏幕作为输入输出设备的数据传递过程称为标准输入输出。流是一个抽象的概念,进行I/O操作时,必须将流与一种具体的物理设备联系起来。10.2.1 标准输入输出流类标准流是不需要打开和关闭文件即可直接操作的流式文件。命名空间std的iostream类库中包括四个标准输入输出流对象的说明。cin:类istream的对象,用来处理标准输入,即键盘输入。cout:类ostream的对象,用来处理标准输出,即屏幕输出。cerr和c
6、log:类ostream的对象,与错误信息的标准输出设备(屏幕)相关联,前者为非缓冲方式,后者为缓冲方式。1标准输入流 cin是由输入类istream的派生类istream_withassign定义的对象,在缺省的情况下,cin所关联的外部设备为键盘,实现从键盘上输入数据。cin为缓冲流。2标准输出流在C+流类体系中定义的标准输出流是cout、cerr、clog,是由输出类ostream的派生类ostream_withassign定义的对象,在缺省的情况下,cout、cerr、clog 所关联的外部设备为屏幕,实现数据流输出到屏幕。cout、clog为缓冲流,而cerr为非缓冲流。【例10.1
7、】输入x,输出x2。/*ex10_1.cpp*#include using namespace std;void main()float x;cerrx;clogx*x=x*xendl;10.2.2 数据输入输出成员函数 1数据输入成员函数(1)字符输入成员函数成员函数get()可以从输入流中获取字符,并将它存放在指定的变量中。该函数有以下两种格式:vch=cin.get()vcin.get(ch)函数实现从输入流中读取一个字符,赋给字符变量ch。采用第2种格式时,如果读取数据成功则函数返回非0值(真),否则(遇结束标志符)返回0值(假)。【例10.2】读取字符#include using n
8、amespace std;void main()char c1,c2,c3;cout输入字符:;c1=cin.get();cin.get(c2);cin.get(c3);coutc1=c1endl;coutc2=c2endl;coutc3=c3endl;输入字符:a b cc1=ac2=c3=b(2)字符串输入成员函数从输入流中一次读取一串字符,有以下2种方式:vcin.get(字符数组或字符指针,字符个数n,终止字符)vcin.getline(字符数组或字符指针,字符个数n,终止字符)从输入流中读取n1个字符,赋给指定的字符数组(或字符指针指向的数组)。如果在读取n1个字符之前遇到指定的终止
9、字符,则提前结束读取。如果读取成功则函数返回非0值(真),否则(遇文件结束符)返回0值(假)。第三个参数缺省为换行符。【例10.3】读取字符串#include using namespace std;void main()char str20;cout输入一行字符串:;cin.getline(str,20);/也可用 cin.get(str,20);coutstr=strendl;程序运行结果如下:输入1行字符串:Hello,everyone.str=Hello,everyone.2数据输出成员函数(1)字符输出成员函数函数put()用于输出单个字符,格式如下:cout.put(char c)
10、;put函数的参数c可以是字符或字符的ASCII码,例如:cout.put(a);/在屏幕上显示字符a(2)字符串输出成员函数函数write()用于输出一个指定长度的字符串,格式如下:cout.write(字符串,字符个数n);例如:cout.write(This is a book.n,20);10.2.3 格式控制成员函数数据输出的格式控制可通过两种途径实现:使用预定义格式控制符l带参数(如setw(n))的:在头文件iomanip.h中定义l不带参数(如dec)的:在头文件iostream.h中定义使用由ios类定义的格式控制成员函数:在头文件iostream.h中说明1输出宽度和填充字
11、符输出宽度和填充字符可通过由ios类中定义的输出域宽控制成员函数width(n)和填充成员函数fill(c)实现。其中n为输出数据的宽度,c为填充字符。【例10.4】使用成员函数控制输出宽度及填充方式#include using namespace std;void main()double a=1234.56;cout.fill(*);for(int i=0;i4;i+)cout.width(i+6);coutaendl;程序运行后输出:1234.561234.56*1234.56*1234.562其它格式控制若要改变流的其它输出格式(如对齐方式),可使用ios类中定义的成员函数setf来实
12、现,函数格式如下:cout.setf(格式标志)格式标志在类ios中被定义为枚举值,在引用这些格式标志时要在前面加上类名ios和域运算符“:”。【例10.5】将商品名称与单价通过初始化赋给数组mname、price,输出各商品的名称和单价,指定商品名称域宽为10个字符、左对齐,单价为右对齐、用定点方式、小数点后有2位有效数字。分析:浮点数默认的输出精度为6(即输出6位有效数字),例如浮点数3456.7891 显示为3456.79。若要实现指定小数点后显示几位有效数字,应先用setf设置定点方式,再用成员函数precision(n)设置小数点后n位有效数字。#include using name
13、space std;void main()char*mname=pencil,pen,book,pencilbox,bag;double price=1.9,17.2,8.63,24,65;for(int i=0;i5;i+)cout.setf(ios:left);/设置左对齐 cout.width(10);/设置域宽为10 coutmnamei;cout.unsetf(ios:left);cout.setf(ios:fixed);/设置定点方式 cout.precision(2);/设置小数点后2位有效数字 coutpricein;程序运行后,输出如下:pencil 1.90pen 17.2
14、0book 8.63pencilbox 24.00bag 65.0010.3 文件输入输出输入输出10.3.1 文件的概念 1文件文件(File)指存储于外部介质上的信息集合,分为文本文件和二进制文件。文本文件中的每一个字节分别为一个字符对应的ASCII 码;二进制文件则是把数据按其在内存中的存储形式原样输出到磁盘上存放。例如整数1025,以文本形式(4948 5053 H)存储占用四个字节,以二进制形式(0401 H)存储则只占用两个字节。2文件的操作文件操作分成4步:(1)定义文件流对象ifstream 文件输入流对象;ofstream 文件输出流对象;fstream 文件输入/出流对象;
15、(2)打开文件文件流对象.open(磁盘文件名,输入输出模式)也可以在定义文件流对象时指定文件打开模式,在构造过程中打开该文件。定义方式如下:ifstream 文件流对象(磁盘文件名,输入输出模式);ofstream 文件流对象(磁盘文件名,输入输出模式);fstream 文件流对象(磁盘文件名,输入输出模式);(3)读/写文件对文件的操作是通过调用其父类ios、istream、ostream中说明的成员函数来实现的(4)关闭文件文件流对象.close();10.3.2 文本文件的读写1使用提取运算符或插入运算符对文件进行读写操作。【例10.6】使用提取运算符和插入运算符将源文件中的前100个
16、字符复制到目标文件中。#include#include using namespace std;void main()char fname120,fname220;coutfname1;ifstream infile(fname1);/定义文件输入流对象,打开源文件 if(!infile)cout不能打开源文件:fname1endl;return;coutfname2;ofstream outfile(fname2);/定义文件输出流对象,打开目标文件 char ch;int i;i=0;while(ich;/从源文件中提取一个字符到变量ch中;outfilech;/将ch中的字符写入目标文件
17、中。i=i+1;infile.close();/关闭源文件 outfile.close();/关闭目标文件2使用成员函数进行文件的读写操作【例10.7】使用成员函数get与put将源文件的内容复制到目标文件中分析:先打开源文件和目标文件,依次从源文件中读取一个字符,并将所读字符写入目标文件中,直到源文件中所有字符读完为止。#include#include using namespace std;void main()char fname120,fname220;coutfname1;ifstream infile;/定义文件输入流对象 infile.open(fname1,ios:in);/打
18、开源文件 if(!infile)cout源文件不存在!fname1endl;return;coutfname2;ofstream outfile;/定义文件输出流对象 outfile.open(fname2,ios:out);/打开目标文件 char ch;while(infile.get(ch)/从源文件中提取一个字符到变量ch中;outfile.put(ch);/将ch中的字符写入目标文件中。infile.close();outfile.close();10.3.3 二进制文件的读写 1二进制文件的写操作二进制文件的写操作是通过成员函数write()来实现:write(字符数组或字符指针,
19、字节数);2二进制文件的读操作二进制文件的读操作是通过成员函数read()来实现:read(字符数组或字符指针,字节数);【例10.8】把090度的正弦值写到二进制文件bin.dat中#include#include#include using namespace std;void main()fstream outfile(bin.dat,ios:out|ios:binary);int i;if(!outfile)cout不能打开输出文件n;return;double s91;for(i=0;i=90;i+)si=sin(i*3.1415926/180);outfile.write(char
20、*)s,sizeof(double)*91);/一次写入91个实数 outfile.close();3测试文件结束测试二进制文件结束位置可用成员函数eof()实现,当到达文件结束位置时,该函数返回零值,否则返回非零值。4返回读入数据的长度返回最近一次输入所读入的字节数可用成员函数gcount()实现。【例10.10】使用读写二进制的成员函数,实现文件的复制。#include#include using namespace std;int main()char fname120,fname220;char buff4096;/建立4K缓冲区 fstream infile,outfile;cout
21、fname1;infile.open(fname1,ios:in|ios:binary);if(!infile)cout不能打开输入文件:fname1endl;return(1);coutfname2;outfile.open(fname2,ios:out|ios:binary);if(!outfile)cout目标文件fname2已存在!endl;return(2);int n;while(!infile.eof()infile.read(buff,4096);n=infile.gcount();/取实际读的字节数 outfile.write(buff,n);/按实际读的字节数写入文件 in
22、file.close();outfile.close();return(0);10.3.4 文件的随机读写 从文件中任何位置开始进行读或写数据,这种读写方式称为文件的随机访问或直接存取。1文件定位 C+的类库fstream中定义了两个与文件相联系的指针v读指针v写指针 C+的文件定位分为读位置和写位置的定位,对应的成员函数 vseekg():设置读位置vSeekp:设置写位置 1 seekg函数seekg(位移量n);seekg(位移量n,参照位置);第1种格式专用于相对文件头指针移动n个字节,第2种格式用于相对参照位置指针移动n个字节。参照位置可取三个值:io3:beg或0:文件头。io3:
23、cur或1:文件指针当前的位置。io3:end或2:文件尾。2 seekp函数文件写操作的定位是通过成员函数seekp实现的。函数的格式如下:seekp(位移量n);seekp(位移量n,参照位置);第1种格式专用于相对文件头指针移动n个字节,第2种格式用于相对参照位置指针移动n个字节。3.tellg函数返回输入文件中文件指针的当前位置是通过成员函数tellg实现的,该函数的返回值为streampos类型。利用seekg和tellg可以获取文件的大小。例如:ifstream tfile(fname,ios:in|ios:nocreate);tfile.seekg(0,ios:end);long
24、 n=tfile.tellg();cout文件大小为n-1个字节;4.tellp函数返回输出文件中文件指针的当前位置是通过成员函数tellp实现的,该函数的返回值为streampos类型。函数的格式如下:tellp();10.4 字符串流 字符串流则是以内存中用户自定义的字符数组(字符串)为输入输出的对象,因此字符串流又称为内存流。10.4.1 字符串流的概念字符串流对象关联的不是文件,而是内存中的一个字符数组,因此不需打开和关闭流对象。字符串流类包括istrstream、ostrstream和strstream。10.4.2 字符串流的输出操作1字符串输出流的定义定义字符串输出流,即建立存储
25、所插入数据的数组对象的语句格式为:ostrstream 字符串流对象(字符数组,缓冲区大小n,操作模式);strstream 字符串流对象(字符数组,缓冲区大小n,操作模式);2字符串输出流的操作(1)pcount()返回当前字符数组中已经插入的字符个数;(2)str()返回存储在字符数组中的字符串。例10.12】字符串流的输出操作#include#include using namespace std;void main()char buf50;/建立字符串输出流对象,与数组buf关联,缓冲区长50个字节 ostrstream out1(buf,sizeof(buf);float PI=3.
26、14159;out1圆周率为:PI0;/向字符数组中写入字符型、实型数据 coutbufendl;cout已插入out1.pcount()个字符endl;运行程序,输出结果如下:圆周率为:3.14159已插入18个字符10.4.3 字符串流的输入操作1字符串输入流的定义定义字符串输入流,即初始化所创建的字符串输入流对象的语句格式为:istrstream 字符串流对象(字符数组,缓冲区大小n);strstream 字符串流对象(字符数组,缓冲区大小n,ios:in);2字符串输入流的操作一个字符串输入流被定义后,可以调用相应的成员函数进行数据的输入操作。【例10.13】字符串流的输入操作#include#includeusing namespace std;void main()char buf=7890 5.4321;int x;double y;istrstream ss(buf);ssxy;cout.setf(ios:fixed);cout.precision(6);coutx+yendl;运行程序,输出结果如下:7895.432100