- 在线时间
- 0 小时
- 最后登录
- 2007-9-23
- 注册时间
- 2004-9-10
- 听众数
- 3
- 收听数
- 0
- 能力
- 0 分
- 体力
- 9975 点
- 威望
- 7 点
- 阅读权限
- 150
- 积分
- 4048
- 相册
- 0
- 日志
- 0
- 记录
- 0
- 帖子
- 1893
- 主题
- 823
- 精华
- 2
- 分享
- 0
- 好友
- 0

我的地盘我做主
该用户从未签到
 |
< >、基于C的文件操作
; r$ A2 @; @1 ~# l' c 在ANSI C中,对文件的操作分为两种方式,即流式文件操作和I/O文件操作,下面就分别介绍之。</P>
0 y8 w( Z" D# B6 [; J6 Y# r< >一、流式文件操作! V6 {# [+ b: c) ^7 s3 y
这种方式的文件操作有一个重要的结构FILE,FILE在stdio.h中定义如下:</P>+ L* Z: K2 u- X3 ^" [, L4 H8 m
< >typedef struct {
6 M$ z7 x- W# u- X: F2 Eint level; /* fill/empty level of buffer */
4 d, h! k8 S+ tunsigned flags; /* File status flags */5 C' `" v$ e& B4 w9 w
char fd; /* File descriptor */
- N- b1 F2 X; W9 H) Ounsigned char hold; /* Ungetc char if no buffer */
0 C$ |/ Q3 h0 w3 ]3 N- lint bsize; /* Buffer size */
6 Z3 T8 n+ d# s; B/ h Punsigned char _FAR *buffer; /* Data transfer buffer */
# U. ~6 v/ v. I# Z! u! runsigned char _FAR *curp; /* Current active pointer */
/ M" d, B" U9 `9 Qunsigned istemp; /* Temporary file indicator */
2 b8 y2 T4 }+ N# yshort token; /* Used for validity checking */7 ]: z1 H) ]8 U; Q) }; `& W( J
} FILE; /* This is the FILE object */</P>
8 \* `$ I$ S; i; C< > FILE这个结构包含了文件操作的基本属性,对文件的操作都要通过这个结构的指针来进行,此种文件操作常用的函数见下表 函数 功能
( v1 U$ c( J! V4 a& Wfopen() 打开流
5 b2 @+ D& e7 z) Wfclose() 关闭流 2 V2 T# g: _1 n6 O
fputc() 写一个字符到流中 & q( H p' L4 [! g; d3 \ m$ k
fgetc() 从流中读一个字符 ; E$ O2 U, j& f* [9 k
fseek() 在流中定位到指定的字符 2 {* c# F2 ]- X# ]% n/ Z" k
fputs() 写字符串到流
d* B, [! n1 y6 R7 H Zfgets() 从流中读一行或指定个字符
1 @% m3 l2 m& l0 y1 E( D Z0 `fprintf() 按格式输出到流 7 L+ O) W* `: x! ~
fscanf() 从流中按格式读取 ; o% h7 O# f1 C! |3 |
feof() 到达文件尾时返回真值 4 ?# ?$ V0 q/ E$ ~% i
ferror() 发生错误时返回其值 : n! q# e( A4 k
rewind() 复位文件定位器到文件开始处
. @8 l [) D) bremove() 删除文件
4 ?% i5 Z& Z C8 {# c- ^0 Nfread() 从流中读指定个数的字符
* @3 w8 X: p. j6 g1 l8 S5 q0 Mfwrite() 向流中写指定个数的字符
2 f4 P7 d: V% o, L$ N5 p5 Q+ j- Itmpfile() 生成一个临时文件流
# g/ g# f5 X9 T, ~$ B x7 Xtmpnam() 生成一个唯一的文件名 </P>
$ {) H5 t y/ f" s, b( k1 d< >
, z6 i4 k* U5 B+ _ 下面就介绍一下这些函数</P>% ~* `$ D/ {4 P" B
< >1.fopen()) x9 `! w' G# G" m
fopen的原型是:FILE *fopen(const char *filename,const char *mode),fopen实现三个功能</P>
- r$ ^ J! }5 _6 g4 T" ^ T0 r< >为使用而打开一个流 , ?" Y2 y* a9 `" E
把一个文件和此流相连接
, c8 O. X* ]; F给此流返回一个FILR指针6 D8 `" D; l& t8 o! Y
参数filename指向要打开的文件名,mode表示打开状态的字符串,其取值如下表</P>; t0 {8 g" M$ H; n+ Y
< >字符串 含义 7 O( p, O$ V8 {1 a" ^' e+ N
"r" 以只读方式打开文件
* V* R2 y, s" k ~( @/ ]! R"w" 以只写方式打开文件
4 b+ `: V- {3 b/ V2 x+ }& L# A"a" 以追加方式打开文件 . x$ g; | l( k6 U/ k$ M; n
"r+" 以读/写方式打开文件,如无文件出错 $ ~% {' x$ i* \/ F5 B1 H, A
"w+" 以读/写方式打开文件,如无文件生成新文件 </P>( g" r3 f7 {' q3 K$ Z* D
< > 一个文件可以以文本模式或二进制模式打开,这两种的区别是:在文本模式中回车被当成一个字符'\n',而二进制模式认为它是两个字符0x0D,0x0A;如果在文件中读到0x1B,文本模式会认为这是文件结束符,也就是二进制模型不会对文件进行处理,而文本方式会按一定的方式对数据作相应的转换。</P>
+ I2 G# P/ ]9 U" Q< > 系统默认的是以文本模式打开,可以修改全部变量_fmode的值来修改这个设置,例如_fmode=O_TEXT;就设置默认打开方式为文本模式;而_fmode=O_BINARY;则设置默认打开方式是二进制模式。</P>
6 ]6 J. f6 O5 L% }3 L< > 我们也可以在模式字符串中指定打开的模式,如"rb"表示以二进制模式打开只读文件,"w+t"或"wt+"表示以文本模式打开读/写文件。</P>$ ^* e7 \4 q1 w
< > 此函数返回一个FILE指针,所以申明一个FILE指针后不用初始化,而是用fopen()来返回一个指针并与一个特定的文件相连,如果成败,返回NULL。</P>3 d- W8 _9 C3 T1 o1 W3 z3 f% C
< >例:</P>, \' Y R/ k* Y6 D" i. t x( Q9 W
< > FILE *fp; / M9 U, }) G1 @1 M5 j) e& X/ C
if(fp=fopen("123.456","wb"))
$ D$ l+ H& }. ?3 T puts("打开文件成功");3 @7 I- }7 E6 i1 t7 l( ?. `
else
% f) m& c$ k( T. T, W; L- A; o/ m2 Z+ J puts("打开文件成败"); </P>! O- s" P3 v4 J
< >2.fclose() . t# i- w" v6 @
fclose()的功能就是关闭用fopen()打开的文件,其原型是:int fclose(FILE *fp);如果成功,返回0,失败返回EOF。</P>
( a; m, g A$ }# y- d4 x< > 在程序结束时一定要记得关闭打开的文件,不然可能会造成数据丢失的情况,我以前就经常犯这样的毛病。</P>
4 _+ f, b5 t. e/ k< >例:fclose(fp);</P>4 O. ~( v8 n$ }# {% M
< >3.fputc()* `9 S2 D: R" C5 }. Q+ }: C$ b$ r
向流写一个字符,原型是int fputc(int c, FILE *stream); 成功返回这个字符,失败返回EOF。</P>
$ i. K9 l8 I& W) X% \, w< >例:fputc('X',fp);</P>
( ]- n: r5 X' p< >4.fgetc()
5 O5 }9 l/ h: Y$ z& v 从流中读一个字符,原型是int fputc(FILE *stream); 成功返回这个字符,失败返回EOF。</P>
" g* @$ b. w1 i# [! [% D$ @( M" v4 x< >例:char ch1=fgetc(fp);</P>8 L1 P# m& I7 `8 j: Q n, D
< >5. fseek()
+ F, G4 @0 g% B5 A. [ Z8 X 此函数一般用于二进制模式打开的文件中,功能是定位到流中指定的位置,原型是int fseek(FILE *stream, long offset, int whence);如果成功返回0,参数offset是移动的字符数,whence是移动的基准,取值是</P>/ A; u5 c; [+ K) O5 p; b. o
< >符号常量 值 基准位置
5 Z r. P' k# ?4 H3 Z/ `SEEK_SET 0 文件开头
4 P( f; @& p3 wSEEK_CUR 1 当前读写的位置
( M3 G" m% @) p9 }SEEK_END 2 文件尾部 </P>
~ a5 V) ]) T< >例:fseek(fp,1234L,SEEK_CUR);//把读写位置从当前位置向后移动1234字节(L后缀表示长整数)</P>0 H3 j7 @4 d8 {
< > fseek(fp,0L,2);//把读写位置移动到文件尾</P>9 u' K5 }* X& Q8 t
< >6.fputs()* ^' G! h! y( `* R2 E# y1 T5 I& ?
写一个字符串到流中,原型int fputs(const char *s, FILE *stream); </P>( ]; ]2 `. T3 \6 ?3 t- t
< >例:fputs("I Love You",fp);</P>
1 F3 s$ C; t3 c: k$ b v' h7 |< >7.fgets()3 D0 N3 Q+ U; F. U( }1 p% B; t
从流中读一行或指定个字符,原型是char *fgets(char *s, int n, FILE *stream); 从流中读取n-1个字符,除非读完一行,参数s是来接收字符串,如果成功则返回s的指针,否则返回NULL。</P>7 c/ Q7 G1 `( m1 \% u! G1 `1 X
< >例:如果一个文件的当前位置的文本如下</P>, {9 j, o+ l) I. p) L/ Z) H
< >Love ,I Have</P>5 X" j+ K n! Z5 ]; x' y
<P>But ........</P>6 T Y- Q( E/ n5 V* E8 @1 F0 F. h
<P>如果用</P>
& e, Q, l( J' d8 N9 |7 b3 K3 y( v<P> fgets(str1,4,file1);</P>
( V7 E$ ^+ J4 @/ S0 j. x9 r: w<P>则执行后str1="Lov",读取了4-1=3个字符,而如果用</P>
e" T {* D! s8 R! G6 P. W! z<P> fgets(str1,23,file1);</P>
) D: R0 h! v2 |1 K* U p9 m<P>则执行str="Love ,I Have",读取了一行(不包括行尾的'\n')。</P>, {5 ~# Q0 Q" H, H
<P>8.fprintf()
& h7 V, ^3 F/ ?% E 按格式输入到流,其原型是int fprintf(FILE *stream, const char *format[, argument, ...]);其用法和printf()相同,不过不是写到控制台,而是写到流罢了</P>
2 U, n. R; L1 S6 {5 U( d<P>例:fprintf(fp,"%2d%s",4,"Hahaha");</P>
$ j* O* \9 i+ P) r! j<P>9.fscanf()
' y5 W! E4 o+ D9 X 从流中按格式读取,其原型是int fscanf(FILE *stream, const char *format[, address, ...]);其用法和scanf()相同,不过不是从控制台读取,而是从流读取罢了。</P>
* L5 l* C- d) o<P>例:fscanf(fp,"%d%d" ,&x,&y);</P>( j/ N$ q$ {% h# G2 I- S8 @
<P>10.feof()
% F. U2 y, B4 c 检测是否已到文件尾,是返回真,否则返回0,其原型是int feof(FILE *stream);</P>1 o/ B o& S! ?* v
<P>例:if(feof(fp))printf("已到文件尾");</P>* o+ g5 ], x# f# a( t7 s3 \ h0 ^
<P>11.ferror()
! r# d/ {2 p: t, `2 q) Y 原型是int ferror(FILE *stream);返回流最近的错误代码,可用clearerr()来清除它,clearerr()的原型是void clearerr(FILE *stream);</P>
. B2 n- L3 [9 d7 o: l) R& Z<P>例:printf("%d",ferror(fp));</P>
. I; a* [/ J- ?) Z) j<P>12.rewind()
2 }$ [# r' Y0 C6 ]1 I4 X9 k 把当前的读写位置回到文件开始,原型是void rewind(FILE *stream);其实本函数相当于fseek(fp,0L,SEEK_SET);</P>
: D/ L2 F9 C7 X% c<P>例:rewind(fp);</P>8 Q" o3 R; t2 z& V* c! ^) m
<P>12.remove()
; R6 h# c x+ k# K+ p7 Y4 P) z 删除文件,原型是int remove(const char *filename); 参数就是要删除的文件名,成功返回0。</P>3 o- @9 c9 ]8 N0 y2 U. S+ J
<P>例:remove("c:\\io.sys");</P>5 B. X) {% J5 } B2 I
<P>13.fread()" c0 b2 E0 }( I2 O: R, X
从流中读指定个数的字符,原型是size_t fread(void *ptr, size_t size, size_t n, FILE *stream);参数ptr是保存读取的数据,void*的指针可用任何类型的指针来替换,如char*、int *等等来替换;size是每块的字节数;n是读取的块数,如果成功,返回实际读取的块数(不是字节数),本函数一般用于二进制模式打开的文件中。</P>7 K3 O' S/ {7 T0 h, O# r! W
<P>例:</P>
$ z6 q0 X' k6 O0 s4 f<P> char x[4230];- C9 V9 e* I3 v9 ~6 Y5 t3 o
FILE *file1=fopen("c:\\msdos.sys","r");
7 W, W7 s& X3 X fread(x,200,12 ,file1);//共读取200*12=2400个字节</P>0 j. z" }- f+ E
<P>14.fwrite()
3 u6 S8 P/ x* n3 N: z5 s 与fread对应,向流中写指定的数据,原型是size_t fwrite(const void *ptr, size_t size, size_t n, FILE *stream);参数ptr是要写入的数据指针,void*的指针可用任何类型的指针来替换,如char*、int *等等来替换;size是每块的字节数;n是要写的块数,如果成功,返回实际写入的块数(不是字节数),本函数一般用于二进制模式打开的文件中。</P>
4 |& e; f& ?4 j5 U0 n<P>例:</P>- o4 y: B5 j5 E- ^
<P> char x[]="I Love You";
$ f! b% K3 G) G8 R, w4 P. e fwire(x, 6,12,fp);//写入6*12=72字节</P>
) ]. k, G8 T$ {: c. T4 O<P> 将把"I Love"写到流fp中12次,共72字节</P>/ P6 f3 s7 y' y: R4 X w- W
<P>15.tmpfile()9 A; F1 r5 ]6 {
其原型是FILE *tmpfile(void); 生成一个临时文件,以"w+b"的模式打开,并返回这个临时流的指针,如果失败返回NULL。在程序结束时,这个文件会被自动删除。</P>0 o+ F/ _! E" R2 J
<P>例:FILE *fp=tmpfile();</P>2 _! {% e3 l7 d4 O
<P>16.tmpnam();# X: I/ O- k' A4 O0 v# f. T# W
其原型为char *tmpnam(char *s); 生成一个唯一的文件名,其实tmpfile()就调用了此函数,参数s用来保存得到的文件名,并返回这个指针,如果失败,返回NULL。</P>
* H/ y9 [& l, c2 o* B6 l+ O& _8 i& f: I4 q
<P>
( c! l8 ~* x9 D- P, L </P> |
zan
|