C++ ofstream和ifstream详细用法 及文件操作 导入到EXCEL

  把数据写入到EXCEL文件中:

#include "stdafx.h"
#include <iostream>
using namespace std;
#include <stdio.h>
#include <stdlib.h>//包含 exit(0) headfile
int main()

 char sn[]="名字\t年齡\n張3333三\t28\n李四\t30\n"; 
 FILE *fp; 
if((fp=fopen("D:\\result\\SN_MAC\\SN_MAC.xls","w"))==NULL) 
 {  printf("can't write!");  
   exit(0); 
 } 
else 
 {  
 //fwrite(p,sizeof(char),strlen(p),fp); 
 for(int i=0;sn[i];i++)
  fputc(sn[i],fp); 
 } 

 fclose(fp);      //??文件 printf("OK ?行完?\n"); 
 return 0;
}

 

 

 

[导读]

     ofstream是从内存到硬盘,ifstream是从硬盘到内存,其实所谓的流缓冲就是内存空间

    在C++中,有一个stream这个类,所有的I/O都以这个“流”类为基础的,包括我们要认识的文件I/O.

 

    stream这个类有两个重要的运算符:

  1、插入器(<<)

  向流输出数据。比如说系统有一个默认的标准输出流(cout),一般情况下就是指的显示器,所以,cout<<"Write Stdout"<<'\n';就表示把字符串"Write Stdout"和换行字符('\n')输出到标准输出流。

  2、析取器(>>)

  从流中输入数据。比如说系统有一个默认的标准输入流(cin),一般情况下就是指的键盘,所以,cin>>x;就表示从标准输入流中读取一个指定类型的数据。

  在C++中,对文件的操作是通过stream的子类fstream(file stream)来实现的,所以,要用这种方式操作文件,就必须加入头文件fstream.h。

    常见的文件操作:

  一、打开文件

  在fstream类中,有一个成员函数open(),就是用来打开文件的,其原型是:

  void open(const char* filename,int mode,int access);

参数:
  filename:  要打开的文件名
  mode:    要打开文件的方式
  access:   打开文件的属性

  打开文件的方式在类ios(是所有流式I/O类的基类)中定义.

常用的值如下:
  ios::app:   以追加的方式打开文件
  ios::ate:   文件打开后定位到文件尾,ios:app就包含有此属性
  ios::binary: 以二进制方式打开文件,缺省的方式是文本方式。两种方式的区别见前文
  ios::in:    文件以输入方式打开(文件数据输入到内存)
  ios::out:   文件以输出方式打开(内存数据输出到文件)
  ios::nocreate: 不建立文件,所以文件不存在时打开失败
  ios::noreplace:不覆盖文件,所以打开文件时如果文件存在失败
  ios::trunc:  如果文件存在,把文件长度设为0

  可以用“或”把以上属性连接起来,如ios::out|ios::binary


  打开文件的属性取值是:
  0:普通文件,打开访问
  1:只读文件
  2:隐含文件
  4:系统文件

  可以用“或”或者“+”把以上属性连接起来,如3或1|2就是以只读和隐含属性打开文件。

  例如:以二进制输入方式打开文件c:\config.sys
  fstream file1;
  file1.open("c:\\config.sys",ios::binary|ios::in,0);

  如果open函数只有文件名一个参数,则是以读/写普通文件打开,即:
  file1.open("c:\\config.sys"); <=> file1.open("c:\\config.sys",ios::in|ios::out,0);

  另外,fstream还有和open()一样的构造函数,对于上例,在定义的时侯就可以打开文件了:
  fstream file1("c:\\config.sys");  

    特别提出的是,fstream有两个子类:

ifstream(input file stream)和ofstream(outpu file stream),

   ifstream默认以输入方式打开文件

   ofstream默认以输出方式打开文件。
   

输入方式,就是你打开这个文件,然后想做的事情是读取文件的内容
输出方式,就是你打开这个文件,然后想做的事情是往这个文件写东西,然后保存



  ifstream file2("c:\\pdos.def");//以输入方式打开文件
  ofstream file3("c:\\x.123");//以输出方式打开文件  

所以,在实际应用中,根据需要的不同,选择不同的类来定义:

如果想以输入方式打开,就用ifstream来定义;

如果想以输出方式打开,就用ofstream来定义;

如果想以输入/输出方式来打开,就用fstream来定义。


  二、关闭文件
  打开的文件使用完成后一定要关闭,fstream提供了成员函数close()来完成此操作,

如:file1.close();

就把file1相连的文件关闭。

  三、读写文件
  读写文件分为文本文件和二进制文件的读取.

对于文本文件的读取比较简单,用插入器和析取器就可以了;

对于二进制的读取就要复杂些,下要就详细的介绍这两种方式

  1、文本文件的读写
  文本文件的读写很简单:

用插入器(<<)向文件输出;

用析取器(>>)从文件输入。

 

假设file1是以输入方式打开,file2以输出打开。

示例如下:
  file2<<"I Love You";//向文件写入字符串"I Love You"

  int i;
  file1>>i;//从文件输入一个整数值。

  这种方式还有一种简单的格式化能力,比如可以指定输出为16进制等等,具体的格式有以下一些

  操纵符 功能 输入/输出

  dec 格式化为十进制数值数据 输入和输出
  endl 输出一个换行符并刷新此流 输出
  ends 输出一个空字符 输出
  hex 格式化为十六进制数值数据 输入和输出
  oct 格式化为八进制数值数据 输入和输出
  setpxecision(int p) 设置浮点数的精度位数 输出

示例:

#include <iostream> 
#include <fstream> 
using namespace std; 

 

void main( void )
{
//利用ofstream类的构造函数创建一个文件输出流对象来打开文件 
ofstream fout( "d:\\mytest.txt" );  
if ( ! fout)
{
cout << "文件不能打开" <<endl;
}
else
{
// 输出到磁盘文件
fout << "Learning C++ is very useful."<< endl; 
//关闭文件输出流
fout.close();           

//利用ifstream类的构造函数创建一个文件输入流对象
ifstream fin( "d:\\mytest.txt" );   
if ( ! fin)
{
cout << "文件不能打开" <<endl;
}
else
{
char buffer[80];  
            // 从磁盘文件输入
fin >> buffer;  
// 关闭文件输入流      
fin.close();  
cout << buffer << endl;
}
}
ofstream file1;

if (!file1)
{
cout << "文件不能打开" <<endl;
}
else
{
        //open函数只有文件名一个参数,则是以读/写普通文件打开
file1.open("d:\\mytest.txt");
//相当于file1.open("d:\\mytest.txt",ios::out|ios::in,0);

//文本文件的写:用插入器(<<)向文件输出.
        file1<<"写入文件操作!"<<endl;
            file1.close();
}

ifstream file2("d:\\mytest.txt");
if (!file2)
{
cout << "文件不能打开" <<endl;

else
{
char ch1[100];
//文本文件的读:用析取器(>>)从文件输入.
file2>>ch1;

cout<<ch1<<endl;
file2.close();
}
}



  2、二进制文件的读写
  ①put()
  put()函数向流写入一个字符,其原型是ofstream &put(char ch),使用也比较简单,如file1.put('c');就是向流写一个字符'c'。

  ②get()
  get()函数比较灵活,有3种常用的重载形式:

    一种就是和put()对应的形式:ifstream &get(char &ch);功能是从流中读取一个字符,结果保存在引用ch中,如果到文件尾,返回空字符。如file2.get(x);表示从文件中读取一个字符,并把读取的字符保存在x中。

  另一种重载形式的原型是: int get();这种形式是从流中返回一个字符,如果到达文件尾,返回EOF,如x=file2.get();和上例功能是一样的。

  还有一种形式的原型是:ifstream &get(char *buf,int num,char delim='\n');这种形式把字符读入由 buf 指向的数组,直到读入了 num 个字符或遇到了由 delim 指定的字符,如果没使用 delim 这个参数,将使用缺省值换行符'\n'。例如:

  file2.get(str1,127,'A'); //从文件中读取字符到字符串str1,当遇到字符'A'或读取了127个字符时终止。

  ③读写数据块

  要读写二进制数据块,使用成员函数read()和write()成员函数,它们原型如下:

  read(unsigned char *buf,int num);
  write(const unsigned char *buf,int num);

  read()从文件中读取 num 个字符到 buf 指向的缓存中,如果在还未读入 num 个字符时就到了文件尾,可以用成员函数 int gcount();来取得实际读取的字符数;而 write() 从buf 指向的缓存写 num 个字符到文件中,值得注意的是缓存的类型是 unsigned char *,有时可能需要类型转换。

  例:
  unsigned char str1[]="I Love You";
  int n[5];
  ifstream in("xxx.xxx");
  ofstream out("yyy.yyy");
  out.write(str1,strlen(str1));//把字符串str1全部写到yyy.yyy中
  in.read((unsigned char*)n,sizeof(n));//从xxx.xxx中读取指定个整数,注意类型转换
  in.close();out.close(); 四、检测EOF

 

  成员函数eof()用来检测是否到达文件尾,如果到达文件尾返回非0值,否则返回0。原型是int eof();

  例:  if(in.eof()) ShowMessage("已经到达文件尾!");

  五、文件定位

  和C的文件操作方式不同的是,C++ I/O系统管理两个与一个文件相联系的指针。一个是读指针,它说明输入操作在文件中的位置;另一个是写指针,它下次写操作的位置。每次执行输入或输出时,相应的指针自动变化。所以,C++的文件定位分为读位置和写位置的定位,对应的成员函数是seekg()和seekp()。seekg()是设置读位置, seekp是设置写位置。它们最通用的形式如下:

  istream &seekg(streamoff offset,seek_dir origin);
  ostream &seekp(streamoff offset,seek_dir origin);
  streamoff定义于 iostream.h 中,定义有偏移量 offset 所能取得的最大值,seek_dir 表示移动的基准位置,是一个有以下值的枚举:

  ios::beg:  文件开头
  ios::cur:  文件当前位置
  ios::end:  文件结尾

  这两个函数一般用于二进制文件,因为文本文件会因为系统对字符的解释而可能与预想的值不同。例:

  file1.seekg(1234,ios::cur); //把文件的读指针从当前位置向后移1234个字节
  file2.seekp(1234,ios::beg); //把文件的写指针从文件开头向后移1234个字节

 

 

 

c++中输出和输入导屏幕和键盘的类别声明包含再标题文件<iostrream.h>中,而磁盘类文件的 I/O则声明再包含标题文件<fstream.h>内。

输入和输出格式:

输出到磁盘 ofsteam 识别字(“文件名”)

从磁盘读文件 ifsteam 识别字("文件名“)

例如:

ofstream outfile("data.txt"); //写入到磁盘的data.txt中

格式化输入输出:

1 整数数据的输入输出

整数数据存储再磁盘内,每个文字各占一个字节。

例如:

#include <fstream.h>

#include <iostream.h>

#inlude <conio.h>

void main()

{

ofstream outfile("data.txt"); //写入文件

for(int i=0;i<10;i++)

outfile<<i<<" "; //空格是为了避免数值连接在一起加上去的,此文件大小为20字节

cout<<"ok,press a key!";

gerch();
}

程序执行后用记事本打开可以看到数据

0 1 2 3 4 5 6 7 8 9

同样读取文件语句为:

int data;

ifstream infile("data.txt");

for(int i=0;i<10;i++)

{

infile>>data ; //读数据的时候因为数据间有一个空格才能完整的读出,

cout<<data<<" "; //输出到屏幕,加上空格是为了分开数字

}

2 字符数据的输入

字符的输出方式以put(ch)为存入语句,读取语句为get(ch)

例如:

char str[]="hello this is a c++ programe!\n"

"i know this difference \n"

"and mocive";

ofstream outfile ("data.txt");

for(int i=0;i<strlen(Str);i++)

outfile.put(str[i]);

读取:

char ch;

ifstream infile("data.txt");

while(infile)

{

infile.get(ch);

cout<<ch;

}

getch();

}

3 字符串数据的输入

数据多的时候读写速度比较快,输入时以整行字符串加上换行符号一次写入。读取的时候以语句getline(buffer,max),来读取整行数据,直到遇到换行符,每行结尾的\n并不读入,所以在 输出的时候需要加上换行符号,否则数据会连接在一起。

例子:

ofstream outfile("strdata.txt");

outfile<<"use your vote\n";

outfile<<"ouse your weise\n";

读取:

const MAX=80;

char buffer[MAX];

ifstream infile("strdata.txt");

while(infile)

{

infile.getline(buffer,MAX);

cout<<buffer<<endl;

}

4 浮点数

浮点数因为有小数点,在存储数据时与整数相同,只要每个数据加上一个空格就可以区隔相邻的数据

6 二进制文件

上述的格式化文件比较占用硬盘控件,采用二进制存储就可以节约很多控件。它使用write,read()来存储和读取。

ofstream 识别字(”文件名“,ios::binary);

write( 写入地址,写入大小)

ifstream 识别字(”文件名“,ios:binary);

识别字.read(读取地址,读取大小);

例如:infile.read((char*)buffer,sizeof(数据类型));

关闭文件

识别字.close();

例子:

ofstream outfile("data.dat",ios::binary);

for(int i=0;i<100;i++)

{

n[0]=i+1;

outfile.write((Char*)n,sizeof(int)); //还可以用变量和数组为例

// outfile.write((Char*)&i,sizeof(int)); i 为变量

// outfile.write((Char*)n,sizeof(int)); n为数组

}

outfile.close();

ifstream inifile("data.dat",ios:binary);

for(i=0;i<100;i++)

{ inifile.read((Char*)n,sizeof(int);

cout<<n[0]<<" ";

}

7 随机存取文件

文件能够随意读出,读出后又可以更新,更新后可以回存到源文件内。fstream file ;

file.open("文件名",存取模式);

file.open("x.dat",ios::app|ios::in|ios::out|ios::binary);

in 打开文件输入或者读取 ifstream

out 打开文件输出或者写入 ofstream

ate 从文件尾开始写入或者读取

app 加在文件尾

arunc 若文件存在,讲其长度设为0

binary 打开二进制文件

二进制文件中有一个指针,指向当前数据在文件中的位置,这个文件指针和一般的指针变量不一样,它只是一个纯粹的指示器。

函数介绍

seekg()

seekg(0) 指针移到文件的最前面

seekg(0,ios::cur);把当前的指针当作0

seekg(0,ios::end);将指针移到文件尾,若再配合file.tellg()则可以求出文件的大小为多少bytes

以下两个操作都必须在文件关闭后才可以使用

remove("文件名”);把这个文件删除

rename("旧文件名","新文件名");

 

 

 

#include <iostream>
#include <fstream>
using namespace std;

int main()
{
int n[5] = {1, 2, 3, 4, 5};
register int i;

ofstream out("test", ios::out | ios::binary);
if(!out) {
cout << "Cannot open file.\n";
return 1;
}

out.write((char *) &n, sizeof n);

out.close();

for(i = 0; i <5; i++) // clear array
n[i] = 0;

ifstream in("test", ios::in | ios::binary);
if(!in) {
cout << "Cannot open file.\n";
return 1;
}

in.read((char *) &n, sizeof n);

for(i = 0; i <5; i++) // show values read from file
cout << n[i] << " ";

in.close();

return 0;
}

 

 

 

一 打开关闭文件

1 fopen函数 用于打开文件

FILE *fopen(char *filename, *type);

fopen("c:\\ccdos\\clib", "rb");  如果成功的打开一个文件, fopen()函数返回文件指针,否则返回空指针(NULL)。由此可判断文件打开是否成功。

"r" 打开文字文件只读        

"w" 创建文字文件只写          

"a" 增补,如果文件不存在则创建一个

"r+" 打开一个文字文件读/写

"w+" 创建一个文字文件读/写

"a+" 打开或创建一个文件增补

"b" 二进制文件(可以和上面每一项合用)

"t" 文这文件(默认项)

r 以只读方式打开文件,该文件必须存在;
r+ 以可读写方式打开文件,该文件必须存在;
rb+ 读写打开一个二进制文件,允许读数据;
rw+ 读写打开一个文本文件,允许读和写;
w 打开只写文件,若文件存在则文件长度清为0,即该文件内容会消失。若文件不存在则建立该文件;
w+ 打开可读写文件,若文件存在则文件长度清为零,即该文件内容会消失。若文件不存在则建立该文件;
a 以附加的方式打开只写文件。若文件不存在,则会建立该文件,如果文件存在,写入的数据会被加到文件尾,即文件原先的内容会被保留(EOF符保留);
a+ 以附加方式打开可读写的文件。若文件不存在,则会建立该文件,如果文件存在,写入的数据会被加到文件尾后,即文件原先的内容会被保留(原来的EOF符不保留);
wb 只写打开或新建一个二进制文件;只允许写数据;
wb+ 读写打开或建立一个二进制文件,允许读和写;
ab+ 读写打开一个二进制文件,允许读或在文件末追加数据;
at+ 打开一个叫string的文件,a表示append,就是说写入处理的时候是接着原来文件已有内容写入,不是从头写入覆盖掉,t表示打开文件的类型是文本文件,+号表示对文件既可以读也可以写;

2. fclose()函数
fclose()函数用来关闭一个由fopen()函数打开的文件,其调用格式为:

int fclose(FILE *stream);该函数返回一个整型数。当文件关闭成功时,返回0,否则返回一个非零值。可以根据函数的返回值判断文件是否关闭成功。

二 有关文件操作的函数

本节所涉及到的文件读写函数均是指顺序读写,即读写了一条信息后,指针自动加1.

下面分别介绍写操作函数和读操作函数。   

1. 文件的顺序写函数

fprintf()、fputs()和fputc()函数均为文件的顺序写操作函数,其调用格式如下:

int fprintf(FILE *stream, char *format, <variable-list>);

int fputs(char *string, FILE *steam);

int fputc(int  ch, FILE *steam); 上述三个函数的返回值均为整型量。

fprintf() 函数的返回值为实际写入文件中的字符个数(字节数)。如果写错误,则返回一个负数;

fputs()函数返回0时表明将string指针所指的字符串写入文件中的操作成功, 返回非0时,表明写操作失败。

fputc()函数返回一个向文件所写字符的值,此时写操作成功,否则返回EOF(文件结束结束其值为-1, 在stdio.h中定义)表示写操作错误。  

fprintf( ) 函数中格式化的规定与printf( ) 函数相同,所不同的只是fprintf()函数是向文件中写入。而printf()是向屏幕输出。

例如:

fputs("Your score of TOEFLis", fp);/*向所建文件写入一串字符*/
fputc(':', fp);    /*向所建文件写冒号:*/
fprintf(fp, "%d\n", i); /*向所建文件写一整型数*/
fprintf(fp, "%s", s);  /*向所建文件写一字符串*/

2. 文件的顺序读操作函数

  函数fscanf()、fgets()和fgetc()均为文件的顺序读操作函数,其调用格式如下:

int fscanf(FILE *stream, char *format, <address-list>);      

char fgets(char *string, int n, FILE *steam);

int fgetc(FILE *steam);      

fscanf()函数的用法与scanf()函数相似,只是它是从文件中读到信息。 fscanf()函数的返回值为EOF(即-1), 表明读错误,否则读数据成功。读到空白字符自动返回。

fgets()函数从文件中读取至多n-1个字符(n用来指定字符数), 并把它们放入string指向的字符串中,在读入之后自动向字符串未尾加一个空字符,读成功返回string指针,失败返回一个空指针。

fgetc()函数返回文件当前位置的一个字符,读错误时返回EOF。

3. 文件的随机读写

有时用户想直接读取文件中间某处的信息, 若用文件的顺序读写必须从文件 头开始直到要求的文件位置再读, 这显然不方便。Turbo C2.0提供了一组文件的 随机读写函数, 即可以将文件位置指针定位在所要求读写的地方直接读写。

文件的随机读写函数如下:

int fseek (FILE *stream, long offset, int fromwhere);

int fread(void *buf, int size, int count, FILE *stream);

int fwrite(void *buf, int size, int count, FILE *stream);

long ftell(FILE *stream);

fseek()函数的作用是将文件的位置指针设置到从fromwhere开始的第offset 字节的位置上, 其中fromwhere是下列几个宏定义之一:

文件位置指针起始计算位置fromwhere

━━━━━━━━━━━━━━━━━━━━━━━━━━━

符号常数 数值 含义

───────────────────────────

SEEK_SET 0 从文件开头

SEEK_CUR 1 从文件指针的现行位置

SEEK_END 2 从文件末尾

━━━━━━━━━━━━━━━━━━━━━━━━━━━

offset是指文件位置指针从指定开始位置(fromwhere指出的位置)跳过的字节数。它是一个长整型量, 以支持大于64K字节的文件。

fseek()函数一般用于对 二进制文件进行操作。当fseek()函数返回0时表明操作成功, 返回非0表示失败。

fread()函数是从文件中读count个字段, 每个字段长度为size个字节, 并把 它们存放到buf指针所指的缓冲器中。

fwrite()函数是把buf指针所指的缓冲器中, 长度为size个字节的count个字 段写到stream指向的文件中去。

随着读和写字节数的增大, 文件位置指示器也增大, 读多少个字节, 文件位 置指示器相应也跳过多少个字节。读写完毕函数返回所读和所写的字段个数。

ftell()函数返回文件位置指示器的当前值,这个值是指示器从文件头开始 算起的字节数, 返回的数为长整型数, 当返回-1时, 表明出现错误。

下面程序把一个浮点数组以二进制方式写入文件test_b.dat中。

例14:

#include <stdio.h>

main()

{

float f[6]={3.2, -4.34, 25.04, 0.1, 50.56, 80.5};

/*定义浮点数组并初始化*/

int i;

FILE *fp;

fp=fopen("test_b.dat", "wb"); /*创建一个二进制文件只写*/

fwrite(f, sizeof(float), 6, fp);/*将6个浮点数写入文件中*/

fclose(fp); /*关闭文件*/

}

下面例子从test_b.dat文件中读100个整型数, 并把它们放到dat数组中。 例15:

#include <stdio.h>

main()

{

FILE *fp;

intdat[100];

fp=fopen("test_b.dat", "rb");/*打开一个二进制文件只读*/

if(fread(dat, sizeof(int), 100, fp)!=100)

/*判断是否读了100个数*/

{

if(feof(fp))

printf("End of file"); /*不到100个数文件结束*/

else

printf("Read error"); /*读数错误*/

fclose(fp); /*关闭文件*/

}

注意:

当用标准文件函数对文件进行读写操作时, 首先将所读写的内容放进缓冲区, 即写函数只对输出缓冲区进行操作, 读函数只对输入缓冲区进行操作。例如向一 个文件写入内容, 所写的内容将首先放在输出缓冲区中, 直到输出缓冲区存满或使用fclose()函数关闭文件时, 缓冲区的内容才会写入文件中。若无fclose()函数, 则不会向文件中存入所写的内容或写入的文件内容不全。有一个对缓冲区 进行刷新的函数, 即fflush(), 其调用格式为:

int fflush(FILE *stream);该函数将输出缓冲区的内容实际写入文件中, 而将输入缓冲区的内容清除掉。

4. feof()和rewind()函数

这两个函数的调用格式为:

int feof(FILE *stream);

int rewind(FILE *stream);

feof()函数检测文件位置指示器是否到达了文件结尾,若是则返回一个非0值, 否则返回0。这个函数对二进制文件操作特别有用, 因为二进制文件中,文件结尾标志EOF也是一个合法的二进制数,只简单的检查读入字符的值来判断文件是否结束是不行的。如果那样的话, 可能会造成文件未结尾而被认为结尾, 所以就必须有feof()函数。

下面的这条语句是常用的判断文件是否结束的方法。

while(!feof(fp))

fgetc(fp);

rewind()函数用于把文件位置指示器移到文件的起点处, 成功时返回0,否则, 返回非0值。

C语言本身并不提供输入输出语句,输入和输出操作是由函数来实现的。在C标准函数库中提供了一些输入输出函数,例如,printf函数和scanf函数。

基于C++的文件操作

在C++中,有一个stream这个类,所有的I/O都以这个“流”类为基础的,包括我们要认识的文件I/O,stream这个类有两个重要的运算符:

1、插入器(<<) 向流输出数据。比如说系统有一个默认的标准输出流(cout),一般情况下就是指的显示器,所以,cout<<"Write Stdout"<<'\n';就表示把字符串"Write Stdout"和换行字符('\n')输出到标准输出流。

2、析取器(>>) 从流中输入数据。比如说系统有一个默认的标准输入流(cin),一般情况下就是指的键盘,所以,cin>>x;就表示从标准输入流中读取一个指定类型(即变量x的类型)的数据。

在C++中,对文件的操作是通过stream的子类fstream(file stream)来实现的,所以,要用这种方式操作文件,就必须加入头文件fstream.h。下面就把此类的文件操作过程一一道来。

一、打开文件 在fstream类中,有一个成员函数open(),就是用来打开文件的,其原型是:

void open(const char* filename,int openmode,int access);

参数:

filename:  要打开的文件名 mode:    要打开文件的方式
access:   打开文件的属性 打开文件的方式在类ios(是所有流式I/O类的基类)中定义,常用的值如下:

ios::app:   以追加的方式打开文件 ios::ate:   文件打开后定位到文件尾,ios:app就包含有此属性
ios::binary:  以二进制方式打开文件,缺省的方式是文本方式。两种方式的区别见前文 ios::in:    文件以输入方式打开(文件=>程序) ios::out:   文件以输出方式打开 (程序=>文件) ios::nocreate: 不建立文件,所以文件不存在时打开失败  ios::noreplace:不覆盖文件,所以打开文件时如果文件存在失败
ios::trunc:  如果文件存在,把文件长度设为0 可以用“或”把以上属性连接起来,如ios::out|ios::binary

打开文件的属性取值是:

0:普通文件,打开访问 1:只读文件 2:隐含文件 4:系统文件 可以用“或”或者“+”把以上属性连接起来 ,如3或1|2就是以只读和隐含属性打开文件。

例如:以二进制输入方式打开文件c:\config.sys

fstream file1; file1.open("c:\\config.sys",ios::binary|ios::in,0);

如果open函数只有文件名一个参数,则是以读/写普通文件打开,即:

file1.open("c:\\config.sys");<=>file1.open("c:\\config.sys",ios::in|ios::out,0);

另外,fstream还有和open()功能一样的构造函数,对于上例,在定义的时侯就可以打开文件了:

fstream file1("c:\\config.sys");

特别提出的是,fstream有两个子类:ifstream(input file stream)和ofstream(outpu file stream),ifstream默认以输入方式打开文件(文件=>程序),而ofstream默认以输出方式打开文件。

ifstream file2("c:\\pdos.def");//以输入方式打开文件 ofstream file3("c:\\x.123");//以输出方式打开文件

所以,在实际应用中,根据需要的不同,选择不同的类来定义:如果想以输入方式打开,就用ifstream来定义;如果想以输出方式打开,就用ofstream来定义;如果想以输入/输出方式来打开,就用fstream来定义。

二、关闭文件 打开的文件使用完成后一定要关闭,fstream提供了成员函数close()来完成此操作,如:file1.close();就把file1相连的文件关闭。

三、读写文件 读写文件分为文本文件和二进制文件的读取,对于文本文件的读取比较简单,用插入器和析取器就可以了;而对于二进制的读取就要复杂些,下要就详细的介绍这两种方式

1、文本文件的读写 文本文件的读写很简单:用插入器(<<)向文件输出;用析取器(>>)从文件输入。假设file1是以输入方式打开,file2以输出打开。示例如下:

file2<<"I Love You";//向文件写入字符串"I Love You" int i; file1>>i;//从文件输入一个整数值。

这种方式还有一种简单的格式化能力,比如可以指定输出为16进制等等,具体的格式有以下一些

操纵符 功能 输入/输出 dec 格式化为十进制数值数据 输入和输出 endl 输出一个换行符并刷新此流 输出 ends 输出一个空字符 输出 hex 格式化为十六进制数值数据 输入和输出 oct 格式化为八进制数值数据 输入和输出
setpxecision(int p) 设置浮点数的精度位数 输出

比如要把123当作十六进制输出:file1<<<123;要把3.1415926以5位精度输出:file1<<<3.1415926。

2、二进制文件的读写 ①put() put()函数向流写入一个字符,其原型是ofstream &put(char ch),使用也比较简单,如file1.put('c');就是向流写一个字符'c'。

②get() get()函数比较灵活,有3种常用的重载形式:

一种就是和put()对应的形式:ifstream &get(char &ch);功能是从流中读取一个字符,结果保存在引

 

 

在看C++编程思想中,每个练习基本都是使用ofstream,ifstream,fstream,以前粗略知道其用法和含义,在看了几位大牛的博文后,进行整理和总结:

这里主要是讨论fstream的内容:

 
  1. #include <fstream>
  2. ofstream         //文件写操作 内存写入存储设备
  3. ifstream         //文件读操作,存储设备读区到内存中
  4. fstream          //读写操作,对打开的文件可进行读写操作

1.打开文件

在fstream类中,成员函数open()实现打开文件的操作,从而将数据流和文件进行关联,通过ofstream,ifstream,fstream对象进行对文件的读写操作

函数:open()

 
  1. <span style=”font-family:Times New Roman;font-size:16px;”>
  2. public member function
  3. void open ( const char * filename,
  4.             ios_base::openmode mode = ios_base::in | ios_base::out );
  5. void open(const wchar_t *_Filename,
  6.         ios_base::openmode mode= ios_base::in | ios_base::out,
  7.         int prot = ios_base::_Openprot);
  8. </span>

参数: filename   操作文件名

mode        打开文件的方式

prot         打开文件的属性                            //基本很少用到,在查看资料时,发现有两种方式

打开文件的方式在ios类(所以流式I/O的基类)中定义,有如下几种方式:

 

ios::in 为输入(读)而打开文件
ios::out 为输出(写)而打开文件
ios::ate 初始位置:文件尾
ios::app 所有输出附加在文件末尾
ios::trunc 如果文件已存在则先删除该文件
ios::binary 二进制方式

这些方式是能够进行组合使用的,以“或”运算(“|”)的方式:例如

 
  1. ofstream out;
  2. out.open(“Hello.txt”, ios::in|ios::out|ios::binary)                 //根据自己需要进行适当的选取

打开文件的属性同样在ios类中也有定义:

0 普通文件,打开操作
1 只读文件
2 隐含文件
4 系统文件

对于文件的属性也可以使用“或”运算和“+”进行组合使用,这里就不做说明了。

很多程序中,可能会碰到ofstream out(“Hello.txt”), ifstream in(“…”),fstream foi(“…”)这样的的使用,并没有显式的去调用open()函数就进行文件的操作,直接调用了其默认的打开方式,因为在stream类的构造函数 中调用了open()函数,并拥有同样的构造函数,所以在这里可以直接使用流对象进行文件的操作,默认方式如下:

 
  1. <span style=”font-family:Times New Roman;font-size:16px;”>
  2. ofstream out(“…”, ios::out);
  3. ifstream in(“…”, ios::in);
  4. fstream foi(“…”, ios::in|ios::out);
  5. </span>

当使用默认方式进行对文件的操作时,你可以使用成员函数is_open()对文件是否打开进行验证

2.关闭文件

当文件读写操作完成之后,我们必须将文件关闭以使文件重新变为可访问的。成员函数close(),它负责将缓存中的数据排放出来并关闭文件。这个函数一旦被调用,原先的流对象就可以被用来打开其它的文件了,这个文件也就可以重新被其它的进程所访问了。为防止流对象被销毁时还联系着打开的文件,析构函数将会自动调用关闭函数close。

 

3.文本文件的读写

类ofstream, ifstream 和fstream 是分别从ostream, istream 和iostream 中引申而来的。这就是为什么 fstream 的对象可以使用其父类的成员来访问数据。

一般来说,我们将使用这些类与同控制台(console)交互同样的成员函数(cin 和 cout)来进行输入输出。如下面的例题所示,我们使用重载的插入操作符<<:

 
  1.   // writing on a text file
  2.  #include <fiostream.h>
  3.  int main () {
  4.      ofstream out(“out.txt”);
  5.      if (out.is_open())
  6.     {
  7.          out << “This is a line.\n”;
  8.          out << “This is another line.\n”;
  9.          out.close();
  10.      }
  11.      return 0;
  12.  }
  13. //结果: 在out.txt中写入:
  14. This is a line.
  15. This is another line

从文件中读入数据也可以用与 cin>>的使用同样的方法:

 
  1. // reading a text file
  2.    #include <iostream.h>
  3.    #include <fstream.h>
  4.    #include <stdlib.h>
  5.    int main () {
  6.        char buffer[256];
  7.        ifstream in(“test.txt”);
  8.        if (! in.is_open())
  9.        { cout << “Error opening file”; exit (1); }
  10.        while (!in.eof() )
  11.        {
  12.            in.getline (buffer,100);
  13.            cout << buffer << endl;
  14.        }
  15.        return 0;
  16.    }
  17.    //结果 在屏幕上输出
  18.     This is a line.
  19.     This is another line

上面的例子读入一个文本文件的内容,然后将它打印到屏幕上。注意我们使用了一个新的成员函数叫做eof ,它是ifstream 从类 ios 中继承过来的,当到达文件末尾时返回true 。

状态标志符的验证(Verification of state flags)

除了eof()以外,还有一些验证流的状态的成员函数(所有都返回bool型返回值):

  • bad()如果在读写过程中出错,返回 true 。例如:当我们要对一个不是打开为写状态的文件进行写入时,或者我们要写入的设备没有剩余空间的时候。
  • fail()除了与bad() 同样的情况下会返回 true 以外,加上格式错误时也返回true ,例如当想要读入一个整数,而获得了一个字母的时候。
  • eof()如果读文件到达文件末尾,返回true。
  • good()这是最通用的:如果调用以上任何一个函数返回true 的话,此函数返回 false 。

要想重置以上成员函数所检查的状态标志,你可以使用成员函数clear(),没有参数。

获得和设置流指针(get and put stream pointers)

所有输入/输出流对象(i/o streams objects)都有至少一个流指针:

  • ifstream, 类似istream, 有一个被称为get pointer的指针,指向下一个将被读取的元素。
  • ofstream, 类似 ostream, 有一个指针 put pointer ,指向写入下一个元素的位置。
  • fstream, 类似 iostream, 同时继承了get 和 put

我们可以通过使用以下成员函数来读出或配置这些指向流中读写位置的流指针:

  • tellg() 和 tellp()这两个成员函数不用传入参数,返回pos_type 类型的值(根据ANSI-C++ 标准) ,就是一个整数,代表当前get 流指针的位置 (用tellg) 或 put 流指针的位置(用tellp).
  • seekg() 和seekp()这对函数分别用来改变流指针get 和put的位置。两个函数都被重载为两种不同的原型:
    seekg ( pos_type position );
    seekp ( pos_type position );

    使用这个原型,流指针被改变为指向从文件开始计算的一个绝对位置。要求传入的参数类型与函数 tellg 和tellp 的返回值类型相同。

    seekg ( off_type offset, seekdir direction );
    seekp ( off_type offset, seekdir direction );

    使用这个原型可以指定由参数direction决定的一个具体的指针开始计算的一个位移(offset)。它可以是:

    ios::beg 从流开始位置计算的位移
    ios::cur 从流指针当前位置开始计算的位移
    ios::end 从流末尾处开始计算的位移

流指针 get 和 put 的值对文本文件(text file)和二进制文件(binary file)的计算方法都是不同的,因为文本模式的文件中某些特殊字符可能被修改。由于这个原因,建议对以文本文件模式打开的文件总是使用seekg 和 seekp的第一种原型,而且不要对tellg 或 tellp 的返回值进行修改。对二进制文件,你可以任意使用这些函数,应该不会有任何意外的行为产生。

以下例子使用这些函数来获得一个二进制文件的大小:

 
  1. // obtaining file size
  2.    #include <iostream.h>
  3.    #include <fstream.h>
  4.    const char * filename = “test.txt”;
  5.    int main () {
  6.        long l,m;
  7.        ifstream in(filename, ios::in|ios::binary);
  8.        l = in.tellg();
  9.        in.seekg (0, ios::end);
  10.        m = in.tellg();
  11.        in.close();
  12.        cout << “size of ” << filename;
  13.        cout << ” is ” << (m-l) << ” bytes.\n”;
  14.        return 0;
  15.    }
  16.   //结果:
  17.   size of example.txt is 40 bytes.

 

 

4.二进制文件

在二进制文件中,使用<< 和>>,以及函数(如getline)来操作符输入和输出数据,没有什么实际意义,虽然它们是符合语法的。

文件流包括两个为顺序读写数据特殊设计的成员函数:write 和 read。第一个函数 (write) 是ostream 的一个成员函数,都是被ofstream所继承。而read 是istream 的一个成员函数,被ifstream 所继承。类 fstream 的对象同时拥有这两个函数。它们的原型是:

write ( char * buffer, streamsize size );
read ( char * buffer, streamsize size );

这里 buffer 是一块内存的地址,用来存储或读出数据。参数size 是一个整数值,表示要从缓存(buffer)中读出或写入的字符数。

  1. // reading binary file
  2.     #include <iostream>
  3.     #include <fstream.h>
  4.     const char * filename = “test.txt”;
  5.     int main () {
  6.         char * buffer;
  7.         long size;
  8.         ifstream in (filename, ios::in|ios::binary|ios::ate);
  9.         size = in.tellg();
  10.         in.seekg (0, ios::beg);
  11.         buffer = new char [size];
  12.         in.read (buffer, size);
  13.         in.close();
  14.         cout << “the complete file is in a buffer”;
  15.         delete[] buffer;
  16.         return 0;
  17.     }
  18.     //运行结果:
  19.     The complete file is in a buffer

 

5.缓存和同步(Buffers and Synchronization)

当我们对文件流进行操作的时候,它们与一个streambuf 类型的缓存(buffer)联系在一起。这个缓存(buffer)实际是一块内存空间,作为流(stream)和物理文件的媒介。例如,对于一个输出流, 每次成员函数put (写一个单个字符)被调用,这个字符不是直接被写入该输出流所对应的物理文件中的,而是首先被插入到该流的缓存(buffer)中。

当缓存被排放出来(flush)时,它里面的所有数据或者被写入物理媒质中(如果是一个输出流的话),或者简单的被抹掉(如果是一个输入流的话)。这个过程称为同步(synchronization),它会在以下任一情况下发生:

    • 当文件被关闭时: 在文件被关闭之前,所有还没有被完全写出或读取的缓存都将被同步。
    • 当缓存buffer 满时:缓存Buffers 有一定的空间限制。当缓存满时,它会被自动同步。
    • 控制符明确指明:当遇到流中某些特定的控制符时,同步会发生。这些控制符包括:flush 和endl。
    • 明确调用函数sync(): 调用成员函数sync() (无参数)可以引发立即同步。这个函数返回一个int 值,等于-1 表示流没有联系的缓存或操作失败。

 

 

 

 

你可能感兴趣的:(C++)