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

我的地盘我做主
该用户从未签到
 |
< >、基于C的文件操作 . G- }5 H# T- X! e: a' ]
在ANSI C中,对文件的操作分为两种方式,即流式文件操作和I/O文件操作,下面就分别介绍之。</P>0 Q' k5 P! x( ~! s, c
< >一、流式文件操作4 I, ? w9 e/ b( D& j! D) d1 A
这种方式的文件操作有一个重要的结构FILE,FILE在stdio.h中定义如下:</P>- a- }6 m/ z( H- k7 ~* i
< >typedef struct {
) i! ]! v: P3 Cint level; /* fill/empty level of buffer */3 Y5 F, J8 u9 }0 p5 D- ^! N
unsigned flags; /* File status flags */
# i* M1 P: h Q7 Z3 B9 Gchar fd; /* File descriptor */
# m4 R( f0 a" M" l9 bunsigned char hold; /* Ungetc char if no buffer */
. ]6 X0 d( z* xint bsize; /* Buffer size */
) y; {( F. z2 `+ t5 _* W7 C: E) m% hunsigned char _FAR *buffer; /* Data transfer buffer */
3 L, U/ Q. w: F2 X( A/ K. h2 A- u& O* `unsigned char _FAR *curp; /* Current active pointer */( W& x& G% ^& C
unsigned istemp; /* Temporary file indicator */
e' ?/ Q( k- M" |* jshort token; /* Used for validity checking */# e3 y/ g5 f8 z0 Y4 v" q
} FILE; /* This is the FILE object */</P>0 ?% c2 I: F+ S+ a9 c: X; D1 _/ f
< > FILE这个结构包含了文件操作的基本属性,对文件的操作都要通过这个结构的指针来进行,此种文件操作常用的函数见下表 函数 功能
: x% M7 ^. ~" C: Sfopen() 打开流 v( g' S6 M: l( z
fclose() 关闭流 : U3 N( z4 S0 I6 h% o! }! V
fputc() 写一个字符到流中 7 R3 Q1 r7 u6 K* |! D5 t
fgetc() 从流中读一个字符
- Z P1 l- b/ S2 j( ^3 dfseek() 在流中定位到指定的字符
2 ]! n: ~) M' bfputs() 写字符串到流
1 s/ W# A5 V0 `( m$ ufgets() 从流中读一行或指定个字符
: B) @0 q0 @7 ? B# f% u5 `fprintf() 按格式输出到流 1 }" V4 M! e0 x4 R8 S
fscanf() 从流中按格式读取 4 S# H8 m3 T* a' T
feof() 到达文件尾时返回真值 % @. T0 l2 s3 F
ferror() 发生错误时返回其值 - X3 ]0 A' i0 _3 P* x; _8 ^2 f
rewind() 复位文件定位器到文件开始处
6 v" b/ L! ~2 V/ B9 o0 premove() 删除文件 ' z" k+ H/ b9 B1 U- Q6 J/ K' t
fread() 从流中读指定个数的字符 6 _7 C* }$ v4 o) e
fwrite() 向流中写指定个数的字符 t) K0 R$ ~0 q( n
tmpfile() 生成一个临时文件流 7 q5 U9 A D$ x
tmpnam() 生成一个唯一的文件名 </P>
& |; a6 l4 n/ L2 M! J' x< >* w! a2 O& y- R5 U. Q2 s
下面就介绍一下这些函数</P>
@; Y$ y6 E6 r2 C( O< >1.fopen()7 Q" R7 I- `! j y# s+ R2 u
fopen的原型是:FILE *fopen(const char *filename,const char *mode),fopen实现三个功能</P>
) l$ S# V- D7 k' b< >为使用而打开一个流 . {( ^8 }- ^* G# E3 X
把一个文件和此流相连接
4 f. Y% P' f @) x给此流返回一个FILR指针
3 `* C) K, \/ h8 k) D参数filename指向要打开的文件名,mode表示打开状态的字符串,其取值如下表</P>& B0 t8 L9 M8 u2 C
< >字符串 含义
$ H0 f5 ]* v4 A/ J3 x# g/ B0 f"r" 以只读方式打开文件
, }. l5 |( Y5 l( {& i' X* ?"w" 以只写方式打开文件 + c) g! a/ H+ M! z
"a" 以追加方式打开文件
1 f; b6 x# w( v1 `) P"r+" 以读/写方式打开文件,如无文件出错 * l) A" P' P4 c0 m
"w+" 以读/写方式打开文件,如无文件生成新文件 </P>
) u! T5 A7 [0 j< > 一个文件可以以文本模式或二进制模式打开,这两种的区别是:在文本模式中回车被当成一个字符'\n',而二进制模式认为它是两个字符0x0D,0x0A;如果在文件中读到0x1B,文本模式会认为这是文件结束符,也就是二进制模型不会对文件进行处理,而文本方式会按一定的方式对数据作相应的转换。</P>
) f/ F2 k- S: G6 a2 [< > 系统默认的是以文本模式打开,可以修改全部变量_fmode的值来修改这个设置,例如_fmode=O_TEXT;就设置默认打开方式为文本模式;而_fmode=O_BINARY;则设置默认打开方式是二进制模式。</P>; q% q1 j S7 ]
< > 我们也可以在模式字符串中指定打开的模式,如"rb"表示以二进制模式打开只读文件,"w+t"或"wt+"表示以文本模式打开读/写文件。</P>
: `4 O; W, d2 D5 {< > 此函数返回一个FILE指针,所以申明一个FILE指针后不用初始化,而是用fopen()来返回一个指针并与一个特定的文件相连,如果成败,返回NULL。</P>$ Z5 E2 D% F: B2 e3 a) j! F9 ?
< >例:</P>, I7 `6 ~# L8 \7 i7 Q* P& v6 U
< > FILE *fp; : l% I+ k2 G$ i9 k% ~
if(fp=fopen("123.456","wb"))( q# O; C8 C2 q% H/ y+ n1 M
puts("打开文件成功");
) W- l& s! V2 ?; y; x else ! q4 f7 g, @9 m$ U v/ l
puts("打开文件成败"); </P>; R# q& C+ H v/ X* d& B; \$ Q8 `3 }+ P
< >2.fclose() 2 {0 A, u) h q/ \8 B- _
fclose()的功能就是关闭用fopen()打开的文件,其原型是:int fclose(FILE *fp);如果成功,返回0,失败返回EOF。</P>0 U9 K9 U: g* W6 y$ }, a' j
< > 在程序结束时一定要记得关闭打开的文件,不然可能会造成数据丢失的情况,我以前就经常犯这样的毛病。</P>) H) b- ?- I# e* k
< >例:fclose(fp);</P>* O) N, T4 F! B6 F) y
< >3.fputc()
/ [, w0 z7 T% J3 r 向流写一个字符,原型是int fputc(int c, FILE *stream); 成功返回这个字符,失败返回EOF。</P>
8 g4 j& e; J+ u6 z3 y8 E: f# j8 D7 G< >例:fputc('X',fp);</P>
7 m [: V: N4 F: k< >4.fgetc()
* A. s2 @+ I& N4 U: s# |" r 从流中读一个字符,原型是int fputc(FILE *stream); 成功返回这个字符,失败返回EOF。</P>( g* N' K% N' w/ e
< >例:char ch1=fgetc(fp);</P>
1 P6 A1 [9 v' |! N< >5. fseek()
% G9 F+ K, Y j" v& h 此函数一般用于二进制模式打开的文件中,功能是定位到流中指定的位置,原型是int fseek(FILE *stream, long offset, int whence);如果成功返回0,参数offset是移动的字符数,whence是移动的基准,取值是</P>
. w1 p: n7 M0 L) f# r1 [( @< >符号常量 值 基准位置
& }6 b: |6 b$ }4 |( c/ N4 sSEEK_SET 0 文件开头
5 l1 S1 R, X: Y, |SEEK_CUR 1 当前读写的位置
* ?5 C4 Z- }# N1 fSEEK_END 2 文件尾部 </P>
* q9 W9 j& X2 M! ?< >例:fseek(fp,1234L,SEEK_CUR);//把读写位置从当前位置向后移动1234字节(L后缀表示长整数)</P>
9 N) h6 ^. d1 _9 w4 K< > fseek(fp,0L,2);//把读写位置移动到文件尾</P>8 o! p0 g- {( a& C7 _' m" r- p' u/ M
< >6.fputs()
3 s0 w' b5 G3 l7 K5 }8 d3 @) E 写一个字符串到流中,原型int fputs(const char *s, FILE *stream); </P>
& B1 Y2 L! ?) x: ?& u< >例:fputs("I Love You",fp);</P>
3 O ` w& D( D< >7.fgets(): l V1 P7 Y& b; h
从流中读一行或指定个字符,原型是char *fgets(char *s, int n, FILE *stream); 从流中读取n-1个字符,除非读完一行,参数s是来接收字符串,如果成功则返回s的指针,否则返回NULL。</P>
, w, L4 P+ O8 |' a% e% X< >例:如果一个文件的当前位置的文本如下</P>
# z' G0 _* H* t# J< >Love ,I Have</P>
- w) `5 x4 V: r6 P0 n+ i<P>But ........</P>3 `- J* w0 ~! j; C, z* @1 w" A, R6 |* ~
<P>如果用</P>2 u. A1 p8 p( t% P
<P> fgets(str1,4,file1);</P>
7 y8 [3 W+ U( p" [& ?<P>则执行后str1="Lov",读取了4-1=3个字符,而如果用</P>: G7 g7 |7 G4 J
<P> fgets(str1,23,file1);</P>/ _! Y6 [0 ^% F0 m( A, v3 D
<P>则执行str="Love ,I Have",读取了一行(不包括行尾的'\n')。</P>
+ a- m {$ i1 r' _8 D' ]# r<P>8.fprintf()
J+ U# x4 A9 c6 e% }* I 按格式输入到流,其原型是int fprintf(FILE *stream, const char *format[, argument, ...]);其用法和printf()相同,不过不是写到控制台,而是写到流罢了</P>8 o0 l: ^+ a( H( v6 \% g& r
<P>例:fprintf(fp,"%2d%s",4,"Hahaha");</P>6 D9 [. F& E5 s( B3 G* t4 Q1 t6 i
<P>9.fscanf()3 z7 i P9 W, H, U5 Y6 O, L8 Y
从流中按格式读取,其原型是int fscanf(FILE *stream, const char *format[, address, ...]);其用法和scanf()相同,不过不是从控制台读取,而是从流读取罢了。</P>
8 m5 ^+ }7 l! ?5 @& v& i<P>例:fscanf(fp,"%d%d" ,&x,&y);</P>
! u' ]! Y, g0 y+ z9 E4 w8 s ]<P>10.feof()
4 X8 r! k: M3 u b/ c8 K 检测是否已到文件尾,是返回真,否则返回0,其原型是int feof(FILE *stream);</P>7 X: S' b! X4 L$ I
<P>例:if(feof(fp))printf("已到文件尾");</P>
5 H& x! P5 B4 j. P<P>11.ferror()$ i5 n& ?& v( u4 p! A' P5 u
原型是int ferror(FILE *stream);返回流最近的错误代码,可用clearerr()来清除它,clearerr()的原型是void clearerr(FILE *stream);</P>
/ e: j, U2 R1 n" A; G6 h<P>例:printf("%d",ferror(fp));</P>0 ]& g) |/ Y- v
<P>12.rewind(): | O! `5 T! q, h2 ]8 t
把当前的读写位置回到文件开始,原型是void rewind(FILE *stream);其实本函数相当于fseek(fp,0L,SEEK_SET);</P>
& c, N* r& s% h) X<P>例:rewind(fp);</P>5 i! ^4 o# P" ^( h. G3 y
<P>12.remove()
) n9 `" o4 v, t# J1 V 删除文件,原型是int remove(const char *filename); 参数就是要删除的文件名,成功返回0。</P>. y5 N8 E- M; ~ N2 Z6 X( T8 R7 U1 u
<P>例:remove("c:\\io.sys");</P>
$ M' u1 y- n: I) J8 d/ }<P>13.fread()# K/ o E; X0 M! l1 R
从流中读指定个数的字符,原型是size_t fread(void *ptr, size_t size, size_t n, FILE *stream);参数ptr是保存读取的数据,void*的指针可用任何类型的指针来替换,如char*、int *等等来替换;size是每块的字节数;n是读取的块数,如果成功,返回实际读取的块数(不是字节数),本函数一般用于二进制模式打开的文件中。</P>
( \. d$ k) b2 X5 l* o( b, X<P>例:</P>2 l/ r6 o( i" U: H9 E5 U) O- c7 `
<P> char x[4230];
6 E) I! y8 J, u) Y$ B: q B( L FILE *file1=fopen("c:\\msdos.sys","r");
3 p+ e5 {8 J$ x2 t/ ] fread(x,200,12 ,file1);//共读取200*12=2400个字节</P>3 p. \. d$ h" U, T6 z% S6 t# x! D( V
<P>14.fwrite()
" M# k4 P# ~6 a' ?! ?8 J" W 与fread对应,向流中写指定的数据,原型是size_t fwrite(const void *ptr, size_t size, size_t n, FILE *stream);参数ptr是要写入的数据指针,void*的指针可用任何类型的指针来替换,如char*、int *等等来替换;size是每块的字节数;n是要写的块数,如果成功,返回实际写入的块数(不是字节数),本函数一般用于二进制模式打开的文件中。</P>3 K& _- q" I" o3 }# w8 ^0 h& U
<P>例:</P>* ?$ \! |9 a9 R2 w [. `$ }
<P> char x[]="I Love You";0 a1 `- D7 a: t, R3 G3 G6 r
fwire(x, 6,12,fp);//写入6*12=72字节</P>' x. Z, ~; i( a" g; W
<P> 将把"I Love"写到流fp中12次,共72字节</P>
$ U* s/ j- v0 f" j w<P>15.tmpfile()$ N; C% s( x3 b0 c
其原型是FILE *tmpfile(void); 生成一个临时文件,以"w+b"的模式打开,并返回这个临时流的指针,如果失败返回NULL。在程序结束时,这个文件会被自动删除。</P>8 s% w O; N! Y" E x
<P>例:FILE *fp=tmpfile();</P>+ M6 W9 x4 [/ S! A) L# [
<P>16.tmpnam();2 e. d, i2 w( K7 W( F& l. Z
其原型为char *tmpnam(char *s); 生成一个唯一的文件名,其实tmpfile()就调用了此函数,参数s用来保存得到的文件名,并返回这个指针,如果失败,返回NULL。</P>+ L. u1 k+ a) z/ f: E8 |& ?
' a2 L) F* }* h5 C5 l0 j4 a. i w
<P>
7 w6 ~9 Q9 j2 M+ k& k' l </P> |
zan
|