用C語言實現(xiàn)文件讀寫操作
創(chuàng)新互聯(lián)服務項目包括龍沙網(wǎng)站建設、龍沙網(wǎng)站制作、龍沙網(wǎng)頁制作以及龍沙網(wǎng)絡營銷策劃等。多年來,我們專注于互聯(lián)網(wǎng)行業(yè),利用自身積累的技術優(yōu)勢、行業(yè)經(jīng)驗、深度合作伙伴關系等,向廣大中小型企業(yè)、政府機構等提供互聯(lián)網(wǎng)行業(yè)的解決方案,龍沙網(wǎng)站推廣取得了明顯的社會效益與經(jīng)濟效益。目前,我們服務的客戶以成都為中心已經(jīng)輻射到龍沙省份的部分城市,未來相信會繼續(xù)擴大服務區(qū)域并繼續(xù)獲得客戶的支持與信任!
二、直接I/O文件操作
這是C提供的另一種文件操作,它是通過直接存/取文件來完成對文件的處理,而上篇所說流式文件操作是通過緩沖區(qū)來進行;流式文件操作是圍繞一個 FILE指針來進行,而此類文件操作是圍繞一個文件的“句柄”來進行,什么是句柄呢?它是一個整數(shù),是系統(tǒng)用來標識一個文件(在WINDOWS中,句柄的概念擴展到所有設備資源的標識)的唯一的記號。此類文件操作常用的函數(shù)如下表,這些函數(shù)及其所用的一些符號在io.h和fcntl.h中定義,在使用時要加入相應的頭文件。
函數(shù) 說明
open() 打開一個文件并返回它的句柄
close() 關閉一個句柄
lseek() 定位到文件的指定位置
read() 塊讀文件
write() 塊寫文件
eof() 測試文件是否結束
filelength() 取得文件長度
rename() 重命名文件
chsize() 改變文件長度
下面就對這些函數(shù)一一說明:
1.open()
打開一個文件并返回它的句柄,如果失敗,將返回一個小于0的值,原型是int open(const char *path, int access [, unsigned mode]); 參數(shù)path是要打開的文件名,access是打開的模式,mode是可選項。表示文件的屬性,主要用于UNIX系統(tǒng)中,在DOS/WINDOWS這個參數(shù)沒有意義。其中文件的打開模式如下表。
符號 含義 符號 含義 符號 含義
O_RDONLY 只讀方式 O_WRONLY 只寫方式 O_RDWR 讀/寫方式
O_NDELAY 用于UNIX系統(tǒng) O_APPEND 追加方式 O_CREAT 如果文件不存在就創(chuàng)建
O_TRUNC 把文件長度截為0 O_EXCL 和O_CREAT連用,如果文件存在返回錯誤 O_BINARY 二進制方式
O_TEXT 文本方式
對于多個要求,可以用"|"運算符來連接,如O_APPEND|O_TEXT表示以文本模式和追加方式打開文件。
例:int handle=open("c:\\msdos.sys",O_BINARY|O_CREAT|O_WRITE)
2.close()
關閉一個句柄,原型是int close(int handle);如果成功返回0
例:close(handle)
3.lseek()
定位到指定的位置,原型是:long lseek(int handle, long offset, int fromwhere);參數(shù)offset是移動的量,fromwhere是移動的基準位置,取值和前面講的fseek()一樣,SEEK_SET:文件首部;SEEK_CUR:文件當前位置;SEEK_END:文件尾。此函數(shù)返回執(zhí)行后文件新的存取位置。
例:
lseek(handle,-1234L,SEEK_CUR);//把存取位置從當前位置向前移動1234個字節(jié)。
x=lseek(hnd1,0L,SEEK_END);//把存取位置移動到文件尾,x=文件尾的位置即文件長度
4.read()
從文件讀取一塊,原型是int read(int handle, void *buf, unsigned len);參數(shù)buf保存讀出的數(shù)據(jù),len是讀取的字節(jié)。函數(shù)返回實際讀出的字節(jié)。
例:char x[200];read(hnd1,x,200);
5.write()
寫一塊數(shù)據(jù)到文件中,原型是int write(int handle, void *buf, unsigned len);參數(shù)的含義同read(),返回實際寫入的字節(jié)。
例:char x[]="I Love You";write(handle,x,strlen(x));
7.eof()
類似feof(),測試文件是否結束,是返回1,否則返回0;原型是:int eof(int handle);
例:while(!eof(handle1)){……};
8.filelength()
返回文件長度,原型是long filelength(int handle);相當于lseek(handle,0L,SEEK_END)
例:long x=filelength(handle);
9.rename()
重命名文件,原型是int rename(const char *oldname, const char *newname); 參數(shù)oldname是舊文件名,newname是新文件名。成功返回0
例:rename("c:\\config.sys","c:\\config.w40");
10.chsize();
改變文件長度,原型是int chsize(int handle, long size);參數(shù)size表示文件新的長度,成功返回0,否則返回-1,如果指定的長度小于文件長度,則文件被截短;如果指定的長度大于文件長度,則在文件后面補''\0''。
例:chsize(handle,0x12345);
;
C語言對文件進行讀取之前需要先打開文件,然后再進行讀寫,讀寫完之后關閉文件。
可以使用兩組函數(shù)實現(xiàn):
一、C語言庫函數(shù)
打開文件:fopen
讀寫(一般對應成對使用):
fgetc---fputc fgets---fputs fread---fwrite
關閉文件:fclose
二、Linux系統(tǒng)函數(shù)
打開文件:open
讀寫(一般對應成對使用):read---write
關閉文件:close
14.freopen(打開文件)
相關函數(shù) fopen,fclose
表頭文件 #includestdio.h
定義函數(shù) FILE * freopen(const char * path,const char * mode,FILE * stream);
函數(shù)說明 參數(shù)path字符串包含欲打開的文件路徑及文件名,參數(shù)mode請參考fopen()說明。參數(shù)stream為已打開的文件指針。Freopen()會將原stream所打開的文件流關閉,然后打開參數(shù)path的文件。
返回值 文件順利打開后,指向該流的文件指針就會被返回。如果文件打開失敗則返回NULL,并把錯誤代碼存在errno 中。
范例
復制代碼代碼如下:
#includestdio.h
main()
{
FILE * fp;
fp=fopen(“/etc/passwd”,”r”);
fp=freopen(“/etc/group”,”r”,fp);
fclose(fp);
}
15.fseek(移動文件流的讀寫位置)
相關函數(shù) rewind,ftell,fgetpos,fsetpos,lseek
表頭文件 #includestdio.h
定義函數(shù) int fseek(FILE * stream,long offset,int whence);
函數(shù)說明 fseek()用來移動文件流的讀寫位置。參數(shù)stream為已打開的文件指針,參數(shù)offset為根據(jù)參數(shù)whence來移動讀寫位置的位移數(shù)。
參數(shù) whence為下列其中一種:
SEEK_SET從距文件開頭offset位移量為新的讀寫位置。SEEK_CUR 以目前的讀寫位置往后增加offset個位移量。
SEEK_END將讀寫位置指向文件尾后再增加offset個位移量。
當whence值為SEEK_CUR 或SEEK_END時,參數(shù)offset允許負值的出現(xiàn)。
下列是較特別的使用方式:
1) 欲將讀寫位置移動到文件開頭時:fseek(FILE *stream,0,SEEK_SET);
2) 欲將讀寫位置移動到文件尾時:fseek(FILE *stream,0,0SEEK_END);
返回值 當調用成功時則返回0,若有錯誤則返回-1,errno會存放錯誤代碼。
附加說明 fseek()不像lseek()會返回讀寫位置,因此必須使用ftell()來取得目前讀寫的位置。
范例
復制代碼代碼如下:
#includestdio.h
main()
{
FILE * stream;
long offset;
fpos_t pos;
stream=fopen(“/etc/passwd”,”r”);
fseek(stream,5,SEEK_SET);
printf(“offset=%d/n”,ftell(stream));
rewind(stream);
fgetpos(stream,pos);
printf(“offset=%d/n”,pos);
pos=10;
fsetpos(stream,pos);
printf(“offset = %d/n”,ftell(stream));
fclose(stream);
}
執(zhí)行 offset = 5
offset =0
offset=10
16.ftell(取得文件流的讀取位置)
相關函數(shù) fseek,rewind,fgetpos,fsetpos
表頭文件 #includestdio.h
定義函數(shù) long ftell(FILE * stream);
函數(shù)說明 ftell()用來取得文件流目前的讀寫位置。參數(shù)stream為已打開的文件指針。
返回值 當調用成功時則返回目前的讀寫位置,若有錯誤則返回-1,errno會存放錯誤代碼。
錯誤代碼 EBADF 參數(shù)stream無效或可移動讀寫位置的文件流。
范例 參考fseek()。
17.fwrite(將數(shù)據(jù)寫至文件流)
相關函數(shù) fopen,fread,fseek,fscanf
表頭文件 #includestdio.h
定義函數(shù) size_t fwrite(const void * ptr,size_t size,size_t nmemb,FILE * stream);
函數(shù)說明 fwrite()用來將數(shù)據(jù)寫入文件流中。參數(shù)stream為已打開的文件指針,參數(shù)ptr 指向欲寫入的數(shù)據(jù)地址,總共寫入的字符數(shù)以參數(shù)size*nmemb來決定。Fwrite()會返回實際寫入的nmemb數(shù)目。
返回值 返回實際寫入的nmemb數(shù)目。
范例
復制代碼代碼如下:
#includestdio.h
#define set_s (x,y) {strcoy(s[x].name,y);s[x].size=strlen(y);}
#define nmemb 3
struct test
{
char name[20];
int size;
}s[nmemb];
main()
{
FILE * stream;
set_s(0,”Linux!”);
set_s(1,”FreeBSD!”);
set_s(2,”Windows2000.”);
stream=fopen(“/tmp/fwrite”,”w”);
fwrite(s,sizeof(struct test),nmemb,stream);
fclose(stream);
}
執(zhí)行 參考fread()。
18.getc(由文件中讀取一個字符)
相關函數(shù) read,fopen,fread,fgetc
表頭文件 #includestdio.h
定義函數(shù) int getc(FILE * stream);
函數(shù)說明 getc()用來從參數(shù)stream所指的文件中讀取一個字符。若讀到文件尾而無數(shù)據(jù)時便返回EOF。雖然getc()與fgetc()作用相同,但getc()為宏定義,非真正的函數(shù)調用。
返回值 getc()會返回讀取到的字符,若返回EOF則表示到了文件尾。
范例 參考fgetc()。
19.getchar(由標準輸入設備內讀進一字符)
相關函數(shù) fopen,fread,fscanf,getc
表頭文件 #includestdio.h
定義函數(shù) int getchar(void);
函數(shù)說明 getchar()用來從標準輸入設備中讀取一個字符。然后將該字符從unsigned char轉換成int后返回。
返回值 getchar()會返回讀取到的字符,若返回EOF則表示有錯誤發(fā)生。
附加說明 getchar()非真正函數(shù),而是getc(stdin)宏定義。
范例
復制代碼代碼如下:
#includestdio.h
main()
{
FILE * fp;
int c,i;
for(i=0li5;i++)
{
c=getchar();
putchar(c);
}
}
執(zhí)行 1234 /*輸入*/
1234 /*輸出*/
20.gets(由標準輸入設備內讀進一字符串)
相關函數(shù) fopen,fread,fscanf,fgets
表頭文件 #includestdio.h
定義函數(shù) char * gets(char *s);
函數(shù)說明 gets()用來從標準設備讀入字符并存到參數(shù)s所指的內存空間,直到出現(xiàn)換行字符或讀到文件尾為止,最后加上NULL作為字符串結束。
返回值 gets()若成功則返回s指針,返回NULL則表示有錯誤發(fā)生。
附加說明 由于gets()無法知道字符串s的大小,必須遇到換行字符或文件尾才會結束輸入,因此容易造成緩沖溢出的安全性問題。建議使用fgets()取代。
范例 參考fgets()
21.mktemp(產(chǎn)生唯一的臨時文件名)
相關函數(shù) tmpfile
表頭文件 #includestdlib.h
定義函數(shù) char * mktemp(char * template);
函數(shù)說明 mktemp()用來產(chǎn)生唯一的臨時文件名。參數(shù)template所指的文件名稱字符串中最后六個字符必須是XXXXXX。產(chǎn)生后的文件名會借字符串指針返回。
返回值 文件順利打開后,指向該流的文件指針就會被返回。如果文件打開失敗則返回NULL,并把錯誤代碼存在errno中。
附加說明 參數(shù)template所指的文件名稱字符串必須聲明為數(shù)組,如:
char template[ ]=”template-XXXXXX”;
不可用char * template=”template-XXXXXX”;
范例
復制代碼代碼如下:
#includestdlib.h
main()
{
char template[ ]=”template-XXXXXX”;
mktemp(template);
printf(“template=%s/n”,template);
}
22.putc(將一指定字符寫入文件中)
相關函數(shù) fopen,fwrite,fscanf,fputc
表頭文件 #includestdio.h
定義函數(shù) int putc(int c,FILE * stream);
函數(shù)說明 putc()會將參數(shù)c轉為unsigned char后寫入?yún)?shù)stream指定的文件中。雖然putc()與fputc()作用相同,但putc()為宏定義,非真正的函數(shù)調用。
返回值 putc()會返回寫入成功的字符,即參數(shù)c。若返回EOF則代表寫入失敗。
范例 參考fputc()。
23.putchar(將指定的字符寫到標準輸出設備)
相關函數(shù) fopen,fwrite,fscanf,fputc
表頭文件 #includestdio.h
定義函數(shù) int putchar (int c);
函數(shù)說明 putchar()用來將參數(shù)c字符寫到標準輸出設備。
返回值 putchar()會返回輸出成功的字符,即參數(shù)c。若返回EOF則代表輸出失敗。
附加說明 putchar()非真正函數(shù),而是putc(c,stdout)宏定義。
范例 參考getchar()。
24.rewind(重設文件流的讀寫位置為文件開頭)
相關函數(shù) fseek,ftell,fgetpos,fsetpos
表頭文件 #includestdio.h
定義函數(shù) void rewind(FILE * stream);
函數(shù)說明 rewind()用來把文件流的讀寫位置移至文件開頭。參數(shù)stream為已打開的文件指針。此函數(shù)相當于調用fseek(stream,0,SEEK_SET)。
返回值
范例 參考fseek()
25.setbuf(設置文件流的緩沖區(qū))
相關函數(shù) setbuffer,setlinebuf,setvbuf
表頭文件 #includestdio.h
定義函數(shù) void setbuf(FILE * stream,char * buf);
函數(shù)說明 在打開文件流后,讀取內容之前,調用setbuf()可以用來設置文件流的緩沖區(qū)。參數(shù)stream為指定的文件流,參數(shù)buf指向自定的緩沖區(qū)起始地址。如果參數(shù)buf為NULL指針,則為無緩沖IO。Setbuf()相當于調用:setvbuf(stream,buf,buf?_IOFBF:_IONBF,BUFSIZ)
返回值
26.setbuffer(設置文件流的緩沖區(qū))
相關函數(shù) setlinebuf,setbuf,setvbuf
表頭文件 #includestdio.h
定義函數(shù) void setbuffer(FILE * stream,char * buf,size_t size);
函數(shù)說明 在打開文件流后,讀取內容之前,調用setbuffer()可用來設置文件流的緩沖區(qū)。參數(shù)stream為指定的文件流,參數(shù)buf指向自定的緩沖區(qū)起始地址,參數(shù)size為緩沖區(qū)大小。
返回值
27.setlinebuf(設置文件流為線性緩沖區(qū))
相關函數(shù) setbuffer,setbuf,setvbuf
表頭文件 #includestdio.h
定義函數(shù) void setlinebuf(FILE * stream);
函數(shù)說明 setlinebuf()用來設置文件流以換行為依據(jù)的無緩沖IO。相當于調用:setvbuf(stream,(char * )NULL,_IOLBF,0);請參考setvbuf()。
返回值
28.setvbuf(設置文件流的緩沖區(qū))
相關函數(shù) setbuffer,setlinebuf,setbuf
表頭文件 #includestdio.h
定義函數(shù) int setvbuf(FILE * stream,char * buf,int mode,size_t size);
函數(shù)說明 在打開文件流后,讀取內容之前,調用setvbuf()可以用來設置文件流的緩沖區(qū)。參數(shù)stream為指定的文件流,參數(shù)buf指向自定的緩沖區(qū)起始地址,參數(shù)size為緩沖區(qū)大小,參數(shù)mode有下列幾種
_IONBF 無緩沖IO
_IOLBF 以換行為依據(jù)的無緩沖IO
_IOFBF 完全無緩沖IO。如果參數(shù)buf為NULL指針,則為無緩沖IO。
返回值
29.ungetc(將指定字符寫回文件流中)
相關函數(shù) fputc,getchar,getc
表頭文件 #includestdio.h
定義函數(shù) int ungetc(int c,FILE * stream);
函數(shù)說明 ungetc()將參數(shù)c字符寫回參數(shù)stream所指定的文件流。這個寫回的字符會由下一個讀取文件流的函數(shù)取得。
返回值 成功則返回c 字符,若有錯誤則返回EOF。
復制代碼代碼如下:
#include stdio.h
#include stdlib.h
int main()
{
FILE *fp = NULL;
char* str;
char re;
int num = 10;
str = (char*)malloc(100);
//snprintf(str, 10,"test: %s", "0123456789012345678");
// printf("str=%s\n", str);
fp = fopen("/local/test.c","a+");
if (fp==NULL){
printf("Fail to open file\n");
}
// ? ? fseek(fp,-1,SEEK_END);
num = ftell(fp);
printf("test file long:%d\n",num);
fscanf(fp,"%s",str);
printf("str = %s\n",str);
printf("test a: %s\n",str);
while ((re=getc(fp))!=EOF){//getc可以用作fgetc用
printf("%c",re);
}
//fread(str,10,10,fp);
fgets(str,100,fp);
printf("test a: %s\n",str);
sprintf(str,"xiewei test is:%s", "ABCDEFGHIGKMNI");
printf("str2=%s\n", str);
// ?fprintf(fp,"%s\n",str);
fwrite(str,2,10,fp);
num = ftell(fp);
if(str!=NULL){
free(str);
}
fclose(fp);
return 0;
}