转自:http://hi.baidu.com/ctralt/blog/item/cde79fec87f841302697911c.html
fstream提供了三个类,用来实现c++对文件的操作。(文件的创建、读、写)。 ifstream -- 从已有的文件读
ofstream -- 向文件写内容
fstream - 打开文件供读写
文件打开模式:
ios::in 读ios::out 写ios::app 从文件末尾开始写ios::binary 二进制模式ios::nocreate 打开一个文件时,如果文件不存在,不创建文件。ios::noreplace 打开一个文件时,如果文件不存在,创建该文件ios::trunc 打开一个文件,然后清空内容ios::ate 打开一个文件时,将位置移动到文件尾
文件指针位置在c++中的用法:
ios::beg 文件头ios::end 文件尾ios::cur 当前位置例子:file.seekg(0,ios::beg); //让文件指针定位到文件开头 file.seekg(0,ios::end); //让文件指针定位到文件末尾 file.seekg(10,ios::cur); //让文件指针从当前位置向文件末方向移动10个字节 file.seekg(-10,ios::cur); //让文件指针从当前位置向文件开始方向移动10个字节 file.seekg(10,ios::beg); //让文件指针定位到离文件开头10个字节的位置
常用的错误判断方法:
good() 如果文件打开成功bad() 打开文件时发生错误eof() 到达文件尾
实例:
一、写入文件
#include <iostream>#include <fstream>using namespace std;
void main(){ ofstream in;in.open("com.txt",ios::trunc); //ios::trunc表示在打开文件前将文件清空,由于是写入,文件不存在则创建int i;char a='a';for(i=1;i<=26;i++)//将26个数字及英文字母写入文件{ if(i<10) { in<<"0"<<i<<"/t"<<a<<"/n"; a++; } else { in<<i<<"/t"<<a<<"/n"; a++; }}in.close();//关闭文件}
打开com.txt,效果如下:
二、读取文件
上面仅仅是将文本写入文件,并没有读取出来。
以下为读取文件的一种方法:将文件每行内容存储到字符串中,再输出字符串
#include <iostream>#include <fstream>using namespace std;
void main(){char buffer[256];fstream out;out.open("com.txt",ios::in);cout<<"com.txt"<<" 的内容如下:"<<endl;while(!out.eof()){ out.getline(buffer,256,'/n');//getline(char *,int,char) 表示该行字符达到256个或遇到换行就结束 cout<<buffer<<endl;}out.close();cin.get();//cin.get() 是用来读取回车键的,如果没这一行,输出的结果一闪就消失了}
逐个字符的读取文件:
#include <iostream>#include <fstream>using namespace std;
void main(){fstream in;char c;in.open("comn.txt",ios::in);while(!in.eof()){ in>>c; cout<<c;}
in.close();cin.get();}
这个方法读取的文件,所有字符都一起显示,不会分行。这里字母z显示两次,是正常的,因为在输出文件最后一个字母z之后,又输出了一次(可以仔细考虑程序代码)。
读取文件某一行内容:
#include <iostream>#include <fstream>#include <string>using namespace std;
int CountLines(char *filename){ifstream ReadFile;int n=0;string tmp;ReadFile.open(filename,ios::in);//ios::in 表示以只读的方式读取文件if(ReadFile.fail())//文件打开失败:返回0{ return 0;}else//文件存在{ while(getline(ReadFile,tmp)) { n++; } return n;}ReadFile.close();}
string ReadLine(char *filename,int line){int lines,i=0;string temp;fstream file;file.open(filename,ios::in);lines=CountLines(filename);if(line<=0){ return "Error 1: 行数错误,不能为0或负数。";}if(file.fail()){ return "Error 2: 文件不存在。";}if(line>lines){ return "Error 3: 行数超出文件长度。";}while(getline(file,temp)&&i<line-1){ i++;}file.close();return temp;}
void main(){int l;char filename[256];cout<<"请输入文件名:"<<endl;cin>>filename;cout<<"/n请输入要读取的行数:"<<endl;cin>>l;cout<<ReadLine(filename,l);cin.get();cin.get();}
很显然,根据以上程序,利用循环,可以逐行读取整个文件内容。
三、统计文件的行数
#include <iostream>#include <fstream>using namespace std;
int CountLines(char *filename){ifstream ReadFile;int n=0;char line[512];ReadFile.open(filename,ios::in);//ios::in 表示以只读的方式读取文件if(ReadFile.fail())//文件打开失败:返回0{ return 0;}else//文件存在{ while(!ReadFile.eof()) { ReadFile.getline(line,512,'/n'); n++; } return n;}
ReadFile.close();}void main(){cout<<"comn.txt的行数为: "<<CountLines("comn.txt")<<endl;cin.get();}
以上程序的设计思路没有问题,但在实际操作的时候会发现统计出的行数与实际不符,原因在于ReadFile.getline(line,512,'/n')这一句:当一行字符超过512或遇到回车之后,行数自动加1.如果换行符在新的一行,返回的结果会比实际多1;如果不在新的一行,返回结果与实际相符。可以修改如下:
#include <iostream>#include <fstream>#include <string>using namespace std;
int CountLines(char *filename){ifstream ReadFile;int n=0;char line[512];string temp;ReadFile.open(filename,ios::in);//ios::in 表示以只读的方式读取文件if(ReadFile.fail())//文件打开失败:返回0{ return 0;}else//文件存在{while(getline(ReadFile,temp)){ n++;} return n;}
ReadFile.close();}void main(){cout<<"comn.txt的行数为: "<<CountLines("comn.txt")<<endl;cin.get();}
四、读取文件数据到数组
#include <iostream> #include <fstream>#include <string>using namespace std;
int CountLines(char *filename)//获取文件的行数{ifstream ReadFile;int n=0;string temp;ReadFile.open(filename,ios::in);//ios::in 表示以只读的方式读取文件if(ReadFile.fail())//文件打开失败:返回0{ return 0;}else//文件存在,返回文件行数{ while(getline(ReadFile,temp)) { n++; } return n;}ReadFile.close();}
void main() { ifstream file; int LINES;char filename[512];cout<<"请输入要打开的文件名:"<<endl;cin>>filename;file.open(filename,ios::in);if(file.fail()){cout<<"文件不存在."<<endl;file.close();cin.get();cin.get();}else//文件存在{LINES=CountLines(filename);int *tc=new int[LINES]; char *t=new char[LINES]; int i=0; while(!file.eof()) //读取数据到数组{ file>>tc[i]; file>>t[i]; i++; } file.close(); //关闭文件for(i=0;i<LINES;i++)//输出数组内容cout<<tc[i]<<"/t"<<t[i]<<endl;cin.get();cin.get();}}
转自:http://hi.baidu.com/lhx0203/blog/item/14f196085ec67f920b7b82dd.html
C++保存数据到文本文档时注意"空格"和"换行"
程序测试读取已保存的文本文档时发现读取数据有误,可能与"空格"和"换行"有关。输出到屏幕时“空格”和“换行”的代码分别是ends和endl;而保存到文件时“空格”不能用ends而用" ",否则该文件无法正常读取。而“换行”则可用“/n”或者“endl”都可。下面是C++保存矩阵对象数据的详细代码:
void matrix::save(char file_name[]) //输出到文件 { ofstream outfile; outfile.open(file_name); if(outfile.fail()) { cout<<"输出文件打开失败!"<<endl; error(); } outfile<<m<<" "<<n<<endl; for(int i=0;i<m;++i) { for(int j=0;j<n;++j) { if(elem[i][j]<0.000000000000001) elem[i][j]=0; outfile<<elem[i][j]<<" "; } outfile<<endl; } outfile.close(); }
读文件:
inFile.open("train.txt"); //打开训练数据文件 for(int i=0;i<prob.l;i++) { /**/ float fvalue; for (int j=0;j<probfeature;j++) { inFile>>fvalue; if(fvalue!=0.0) { x_space[(probfeature+1)*i+j].index=j+1; x_space[(probfeature+1)*i+j].value=fvalue; } } x_space[(probfeature+1)*i+probfeature].index=-1; prob.x[i]=&x_space[(probfeature+1)*i]; if (i<40) //类别标签 prob.y[i]=1; else prob.y[i]=-1; } inFile.close();
二进制方式:
double* pbuffer = NULL;DWORD filesize = 0;FILE *newfile = NULL;FILE *file = NULL;double img[64][64];
for(int i=0;i<64;i++)for(int j=0;j<64;j++){img[i][j] = 1.0;}
fopen_s(&newfile,"E://h" ,"wb");
//if(file!=NULL){
// fread(pbuffer,1,filesize,file);fwrite(img,sizeof(double),64*64,newfile); fclose(newfile);} fopen_s(&file,"E://h","rb");fseek(file,0L,SEEK_END);filesize=ftell(file); pbuffer = (double*)malloc(filesize);
fseek(file,0L,SEEK_SET);
fread(pbuffer,1,filesize,file);
for(int i=0;i<500;i++)cout<<*(pbuffer+i)<<" ";
Matlab
fid=fopen('Data.txt','w');fprintf(fid,'%g/r/n',a);fclose(fid);