舉例說明
C++檔案流:
fstream // 檔案流
ifstream // 輸入檔案流
ofstream // 輸出檔案流
//創建一個文本檔案並寫入信息
//同向螢幕上輸出信息一樣將信息輸出至檔案
#include<iomanip>
#include<fstream>
void main()
{
ofstream f1("d:\\me.txt"); //打開檔案用於寫,若檔案不存在就創建它
if(!f1)return; //打開檔案失敗則結束運行
f1<<setw(20)<<"姓名:"<<"張三"<<endl; //使用插入運算符寫檔案內容
f1<<setw(20)<<"家庭地址:"<<"華盛頓"<<endl;
f1.close(); //關閉檔案
}
運行後打開檔案d:\me.txt,其內容如下:
姓名:張三
家庭地址:華盛頓
檔案操作
打開檔案
檔案名稱
注意路徑名中的斜槓要雙寫,如:
"D:\\MyFiles\\ReadMe.txt"
檔案打開方式選項
ios::in = 0x01, //供讀,檔案不存在則創建(ifstream默認的打開方式)
ios::out = 0x02, //供寫,檔案不存在則創建,若檔案已存在則清空原內容(ofstream默認的打開方式)
ios::ate = 0x04, //檔案不存在時,生成空檔案;檔案不存在,清空原檔案( ofstream打開方式)。如果沒有檔案,打開失敗;如果有檔案,定位到檔案尾,但是不能寫檔案(i fstream打開方式)。
ios::app = 0x08, //供寫,檔案不存在則創建,若檔案已存在則在原檔案內容後寫入新的內容,指針位置總在最後
ios::trunc = 0x10, //在讀寫前先將檔案長度截斷為0(默認)
ios::nocreate = 0x20, //檔案不存在時產生錯誤,常和in或app聯合使用
ios::noreplace = 0x40, //檔案存在時產生錯誤,常和out聯合使用
ios::binary = 0x80 //二進制格式檔案
註://新版本的c++標準,ios::noreplace和ios::nocreate已經無法使用
檔案保護方式選擇項
filebuf::openprot; //默認的兼容共享方式
filebuf::sh_none; //獨占,不共享
filebuf::sh_read; //讀共享
filebuf::sh_write; //寫共享
以上方式僅舊版VC中支持,新版VC在share.h中為Win32項目定義了如下方式
_SH_DENYRW 0x10 /* deny read/write mode*/
_SH_DENYWR 0x20 /* deny write mode */
_SH_DENYRD 0x30 /* deny read mode */
_SH_DENYNO 0x40 /* deny none mode */
_SH_SECURE 0x80 /* secure mode */
示例:fstream a_file("test.dat", ios::in | ios::binary, _SH_DENYRW);
此時使用其他程式打開test.dat將顯示“此檔案已被其他進程占用”
打開檔案的方法
調用構造函式時指定檔案名稱和打開模式
ifstream f("d:\\12.txt",ios::nocreate); //默認以 ios::in 的方式打開檔案,檔案不存在時操作失敗
ofstream f("d:\\12.txt"); //默認以 ios::out的方式打開檔案
fstream f("d:\\12.dat",ios::in|ios::out|ios::binary); //以讀寫方式打開二進制檔案
使用Open成員函式
fstream f;
f.open("d:\\12.txt",ios::out); //利用同一對象對多個檔案進行操作時要用到open函式
檢查是否成功打開
成功:
if(f){...} //對ifstream、ofstream對象可用,fstream對象不可用。
if(f.good()){...}
失敗:
if(!f){...} // !運算符已經重載
if(f.fail()){...}
讀寫操作
使用<<,>>運算符
只能進行文本檔案的讀寫操作,用於二進制檔案可能會產生錯誤。
使用函式成員 get、put、read、write等:
ofstream的成員函式write從記憶體中的一個指定的位置開始輸出固定數目的位元組到指定的流,當流被關聯檔案時,函式write在檔案中從put檔案定位指針指定的位置開始寫入數據
ifstream的成員函式read將固定數目的位元組從一個指定的流輸入到記憶體中指定地址開始的一部分空間中,若關聯檔案,read函式在檔案中從get檔案定位指針指定的位置開始讀取數據
經常和read配合使用的函式是gcount(),用來獲得實際讀取的位元組數。
讀寫二進制檔案注意事項
打開方式中必須指定ios::binary,否則讀寫會出錯
用read\write進行讀寫操作,而不能使用插入、提取運算符進行操作,否則會出錯。
使用eof()函式檢測檔案是否讀結束,使用gcount()獲得實際讀取的位元組數
關閉檔案
使用成員函式close,如:
f.close();
利用析構函式
對象生命期結束時會檢查檔案是否關閉,對沒有關閉的檔案進行關閉操作。
隨機讀寫檔案
ifstream和ofstream都提供了成員函式來重定位檔案定位指針(檔案中下一個被讀取或寫入的位元組號)
在ifstream中 這個成員函式為seekg("seek get");在ofstream中為seekp("seek put")
seekg(絕對位置); //絕對移動, //輸入流操作
seekg(相對位置,參照位置); //相對操作
tellg(); //返回當前指針位置
seekp(絕對位置); //絕對移動, //輸出流操作
seekp(相對位置,參照位置); //相對操作
tellp()和tellg()成員函式分別用來返回當前get和put的指針位置
參照位置:
ios::beg = 0 //相對於檔案頭
ios::cur = 1 //相對於當前位置
ios::end = 2 //相對於檔案尾
讀寫文本檔案的示例:
//為能夠正確讀出寫入檔案的各數據,各數據間最好要有分隔
#include<fstream>
void main()
{
fstream f("d:\\try.txt",ios::out);
f<<1234<<' '<<3.14<<'A'<<"How are you"; //寫入數據
f.close();
f.open("d:\\try.txt",ios::in);
int i;
double d;
char c;
char s[20];
f>>i>>d>>c; //讀取數據
f.getline(s,20);
cout<<i<<endl; //顯示各數據
cout<<d<<endl;
cout<<c<<endl;
cout<<s<<endl;
f.close();
}
運行結果:
1234
3.14
A
How are you
Press any key to continue
顯示文本檔案的內容
//使用get()一次讀一個位元組--------------------------------方案一
#include<fstream>
void main()
{
ifstream fin("d:\\簡介.txt",ios::nocreate);
if(!fin){
cout<<"File open error!\n";
return;
}
char c;
while((c=fin.get())!=EOF)cout<<c; //注意結束條件的判斷
fin.close();
}
//使用get(char *,int n,char delim="\n")一次讀多個字元----方案二
//巧妙利用文本檔案中不會有字元'\0'的特點進行讀取
#include<fstream>
void main()
{
ifstream fin("d:\\簡介.txt",ios::nocreate);
if(!fin){
cout<<"File open error!\n";
return;
}
char c[80];
while(fin.get(c,80,'\0')!=NULL)cout<<c; //注意結束條件的判斷
fin.close();
}
//使用read(char *,int n)讀檔案---------------------------方案三
#include<fstream>
void main()
{
ifstream fin("d:\\簡介.txt",ios::nocreate);
if(!fin){
cout<<"File open error!\n";
return;
}
char c[80];
while(!fin.eof()) //判斷檔案是否讀結束
{
fin.read(c,80);
cout.write(c,fin.gcount());
}
fin.close();
}
拷貝檔案
//二進制檔案操作示例
#include<fstream>
void main()
{
ifstream fin("C:\\1.exe",ios::nocreate|ios::binary);
if(!fin){
cout<<"File open error!\n";
return;
}
ofstream fout("C:\\2.exe",ios::binary);
char c[1024];
while(!fin.eof())
{
fin.read(c,1024);
fout.write(c,fin.gcount());
}
fin.close();
fout.close();
cout<<"Copy over!\n";