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

我的地盘我做主
该用户从未签到
 |
< >、基于C的文件操作
4 S- a! O/ ~: Y# F d2 A 在ANSI C中,对文件的操作分为两种方式,即流式文件操作和I/O文件操作,下面就分别介绍之。</P>
) ^- R7 d# x; K3 n; x7 F< >一、流式文件操作# l6 K" Z9 _" Z) f
这种方式的文件操作有一个重要的结构FILE,FILE在stdio.h中定义如下:</P>( r4 f- Z/ v" p ]" |6 f3 h
< >typedef struct {
. T/ T1 Q7 }8 Y! m; l6 pint level; /* fill/empty level of buffer */, t/ y: L) w [, W; I
unsigned flags; /* File status flags */
( v; t* _8 R& l1 Lchar fd; /* File descriptor */
6 A. \+ R' r+ b5 { w3 B( kunsigned char hold; /* Ungetc char if no buffer */# U0 `& G' D5 i* c
int bsize; /* Buffer size */
' V, o* Q: ?" H7 N1 z* @unsigned char _FAR *buffer; /* Data transfer buffer */
+ z* s: F& u: ]! @9 g% B3 i% xunsigned char _FAR *curp; /* Current active pointer */' R! U. g8 k3 u: l% g
unsigned istemp; /* Temporary file indicator */
5 d: j4 _1 V, h. z) Dshort token; /* Used for validity checking */
2 d* @0 f4 H; l# p) e} FILE; /* This is the FILE object */</P>
4 [* b* {, V9 M< > FILE这个结构包含了文件操作的基本属性,对文件的操作都要通过这个结构的指针来进行,此种文件操作常用的函数见下表 函数 功能
& @* s1 G4 Q9 T( cfopen() 打开流 6 i( B- f' y) z0 ]4 m0 D+ i7 H
fclose() 关闭流 " z& y9 @2 }8 i9 S
fputc() 写一个字符到流中 ( k& G4 X" n% I$ _5 Z
fgetc() 从流中读一个字符 , ?6 V2 G) V3 n6 r7 e$ W8 E
fseek() 在流中定位到指定的字符
8 b# P6 x% f$ N1 K! t' afputs() 写字符串到流 4 C$ M3 o0 S, q U3 ]) g0 m C5 t q
fgets() 从流中读一行或指定个字符 9 F" A* |3 _9 u6 p
fprintf() 按格式输出到流
7 M$ @; k! j6 a" ?* z* t: Vfscanf() 从流中按格式读取 * ]7 _: k0 E5 p( X
feof() 到达文件尾时返回真值
2 P$ _- `' [( E1 C, a: k) \ W Sferror() 发生错误时返回其值 0 ]2 Y7 _3 W9 N, ]5 q
rewind() 复位文件定位器到文件开始处
1 O7 d/ |$ T% R% Lremove() 删除文件
' ]) ~! L/ n, u* o. \fread() 从流中读指定个数的字符
7 l, F% c; m. @; }fwrite() 向流中写指定个数的字符 % P& k6 O! w/ X4 N3 ?% ^8 K
tmpfile() 生成一个临时文件流 0 {7 ^& D8 P$ `7 m* w2 Y: e
tmpnam() 生成一个唯一的文件名 </P>
7 T6 m0 r* o: E5 ?2 r' W< >$ H5 \6 @* W/ Z7 D; v4 Q/ f; n$ W
下面就介绍一下这些函数</P>- v; R2 d* D' G9 Z3 z( o7 N0 F7 y
< >1.fopen()
0 h, q' f1 Y i: I: c fopen的原型是:FILE *fopen(const char *filename,const char *mode),fopen实现三个功能</P>; I" p: O: U7 F6 f7 r, b: M
< >为使用而打开一个流 0 d1 Y+ F% p1 a
把一个文件和此流相连接 * c- u, j/ \4 m+ U9 X
给此流返回一个FILR指针
; ]1 N( p. f! ~1 ^3 s参数filename指向要打开的文件名,mode表示打开状态的字符串,其取值如下表</P>
1 ^) i2 \. F' r# w; R, d5 R, Q1 o" P< >字符串 含义
) L+ W4 \/ {+ X8 O3 x3 I"r" 以只读方式打开文件 2 F2 t0 h( t' b( |2 q6 G3 }1 `+ ]: K
"w" 以只写方式打开文件
; i$ [4 d) j {+ }! s9 ~"a" 以追加方式打开文件 + p# {( U; {7 r9 C( p# p+ \
"r+" 以读/写方式打开文件,如无文件出错
! @. E( @( U) W0 Q"w+" 以读/写方式打开文件,如无文件生成新文件 </P>
( u# ~2 R+ l! r E% ?8 }< > 一个文件可以以文本模式或二进制模式打开,这两种的区别是:在文本模式中回车被当成一个字符'\n',而二进制模式认为它是两个字符0x0D,0x0A;如果在文件中读到0x1B,文本模式会认为这是文件结束符,也就是二进制模型不会对文件进行处理,而文本方式会按一定的方式对数据作相应的转换。</P>+ g" X& K' N% \8 A
< > 系统默认的是以文本模式打开,可以修改全部变量_fmode的值来修改这个设置,例如_fmode=O_TEXT;就设置默认打开方式为文本模式;而_fmode=O_BINARY;则设置默认打开方式是二进制模式。</P>+ v1 L d% ^) X) O5 ~1 u0 o* T$ s
< > 我们也可以在模式字符串中指定打开的模式,如"rb"表示以二进制模式打开只读文件,"w+t"或"wt+"表示以文本模式打开读/写文件。</P>6 `: k( V5 k. L
< > 此函数返回一个FILE指针,所以申明一个FILE指针后不用初始化,而是用fopen()来返回一个指针并与一个特定的文件相连,如果成败,返回NULL。</P>* C* _& I g' U- p8 q; o; E! U2 l
< >例:</P>
. o) z; C, O% E0 H< > FILE *fp; 8 o" E! n7 v4 X( F m9 V
if(fp=fopen("123.456","wb"))
0 q" ?# x6 C7 u# s. Q( L+ j8 D( u puts("打开文件成功");
# H4 @4 T9 P! ]7 U9 K g else 8 z, {& M' t7 Q2 b+ c
puts("打开文件成败"); </P>2 F5 N2 ?% I: p8 C* n" U1 @2 E
< >2.fclose() " W p! b9 c; l6 `& J
fclose()的功能就是关闭用fopen()打开的文件,其原型是:int fclose(FILE *fp);如果成功,返回0,失败返回EOF。</P>
( |+ M* H9 B8 E2 t< > 在程序结束时一定要记得关闭打开的文件,不然可能会造成数据丢失的情况,我以前就经常犯这样的毛病。</P>
: w( e. c: Z8 `< >例:fclose(fp);</P>
0 C# x2 L& ^7 J* @ N8 O8 d# c7 J< >3.fputc(), _# |; m; ?1 {- G! [
向流写一个字符,原型是int fputc(int c, FILE *stream); 成功返回这个字符,失败返回EOF。</P>6 C+ Q. Q& Z+ X9 ]. h, J
< >例:fputc('X',fp);</P>
: b8 N7 X& y5 M5 M' l3 Z5 @, G! d< >4.fgetc(); d. _9 `2 D3 F( v, b/ e
从流中读一个字符,原型是int fputc(FILE *stream); 成功返回这个字符,失败返回EOF。</P>
: }% B9 {& d; c% {2 ~; e0 D< >例:char ch1=fgetc(fp);</P>
4 ~9 B, }. T+ K9 a7 n( ^< >5. fseek()/ j/ g; `: X# A) a
此函数一般用于二进制模式打开的文件中,功能是定位到流中指定的位置,原型是int fseek(FILE *stream, long offset, int whence);如果成功返回0,参数offset是移动的字符数,whence是移动的基准,取值是</P>/ z5 |) |' K7 Z8 s& ?* Y. u
< >符号常量 值 基准位置 " Z7 |+ {' }7 D2 K/ V$ H# `/ j
SEEK_SET 0 文件开头
0 I9 h; L3 j5 P: x; Q5 d0 ]8 o5 \2 fSEEK_CUR 1 当前读写的位置
! X. ^/ f! b9 u7 e% [9 z4 ^SEEK_END 2 文件尾部 </P>% {6 D7 Z7 M$ G/ k5 Q
< >例:fseek(fp,1234L,SEEK_CUR);//把读写位置从当前位置向后移动1234字节(L后缀表示长整数)</P>0 F3 B" c: H% z- ]( C
< > fseek(fp,0L,2);//把读写位置移动到文件尾</P>
5 u. \# @2 V# h; ]< >6.fputs()
& `9 I! k2 X. d6 r' v0 n/ v- ^' S 写一个字符串到流中,原型int fputs(const char *s, FILE *stream); </P>6 J) n6 _9 b! c' q3 D/ P7 W) c
< >例:fputs("I Love You",fp);</P>
3 i8 T9 Z+ C9 e, E2 v< >7.fgets()
% r4 q8 C# G7 j( y" E 从流中读一行或指定个字符,原型是char *fgets(char *s, int n, FILE *stream); 从流中读取n-1个字符,除非读完一行,参数s是来接收字符串,如果成功则返回s的指针,否则返回NULL。</P>
& Z* Q7 |( E4 E5 k1 |' P7 W Z< >例:如果一个文件的当前位置的文本如下</P>0 k" A, X! t) X$ k) a ^. C
< >Love ,I Have</P>
, H6 _7 {' S6 C' H1 { [9 x<P>But ........</P>
0 w0 `+ d$ V/ L! W J<P>如果用</P>
$ g6 T: e1 t9 c+ V' g7 d6 ^4 A<P> fgets(str1,4,file1);</P>: C3 r+ {- ]' {' B& I
<P>则执行后str1="Lov",读取了4-1=3个字符,而如果用</P>& Z* _4 T# m+ I3 b, J$ j/ ^
<P> fgets(str1,23,file1);</P>
' U5 s0 a: z1 ~( l. e<P>则执行str="Love ,I Have",读取了一行(不包括行尾的'\n')。</P>
" }# Q. P! i8 D3 D# j<P>8.fprintf()
& o$ Z6 E; w& q7 L) t O 按格式输入到流,其原型是int fprintf(FILE *stream, const char *format[, argument, ...]);其用法和printf()相同,不过不是写到控制台,而是写到流罢了</P>* `- e. O3 W. t& k
<P>例:fprintf(fp,"%2d%s",4,"Hahaha");</P>
# @& t% \1 z% ]7 O<P>9.fscanf()7 ^# i5 w( {% ^, n* K5 M
从流中按格式读取,其原型是int fscanf(FILE *stream, const char *format[, address, ...]);其用法和scanf()相同,不过不是从控制台读取,而是从流读取罢了。</P>) W$ ^, S O9 \9 H1 c4 I( a
<P>例:fscanf(fp,"%d%d" ,&x,&y);</P>
1 p" c# |" H: r# {1 f<P>10.feof()
' M6 V/ t$ L; F/ r 检测是否已到文件尾,是返回真,否则返回0,其原型是int feof(FILE *stream);</P>6 r* I" @6 c9 ~' F' w
<P>例:if(feof(fp))printf("已到文件尾");</P>
( a# ? }- ], M<P>11.ferror()
7 |6 W1 u$ B0 }# S1 w8 u/ m 原型是int ferror(FILE *stream);返回流最近的错误代码,可用clearerr()来清除它,clearerr()的原型是void clearerr(FILE *stream);</P>7 h# U5 f4 z9 }
<P>例:printf("%d",ferror(fp));</P>$ F/ }+ u" d! J# V! j2 j" E: o
<P>12.rewind()
, a5 I7 k# |# r9 ^) t- t8 r& ^3 r( @ 把当前的读写位置回到文件开始,原型是void rewind(FILE *stream);其实本函数相当于fseek(fp,0L,SEEK_SET);</P>7 y" o+ T; n' d3 G9 y
<P>例:rewind(fp);</P>
7 z. c3 m- Q2 J3 v m! ?1 G<P>12.remove()
" T# e/ a2 H. s" V* i5 H6 S3 z 删除文件,原型是int remove(const char *filename); 参数就是要删除的文件名,成功返回0。</P>
! }; N1 `; |; @: Q- U, R<P>例:remove("c:\\io.sys");</P>
" c% }, Y: r8 B" D( n0 K9 g* k7 |<P>13.fread()
7 r0 G# }2 s4 K% @' q) m 从流中读指定个数的字符,原型是size_t fread(void *ptr, size_t size, size_t n, FILE *stream);参数ptr是保存读取的数据,void*的指针可用任何类型的指针来替换,如char*、int *等等来替换;size是每块的字节数;n是读取的块数,如果成功,返回实际读取的块数(不是字节数),本函数一般用于二进制模式打开的文件中。</P>
) A/ Z& [2 T, \: t<P>例:</P>; I' |( X2 r" [, [4 o
<P> char x[4230];
1 y; n0 s" Y. v- a+ u+ m" K FILE *file1=fopen("c:\\msdos.sys","r");
' C1 o: ?5 s& m fread(x,200,12 ,file1);//共读取200*12=2400个字节</P>
1 N% @# d7 F2 K. w6 g v<P>14.fwrite()
5 n7 b; C; a6 M0 a g 与fread对应,向流中写指定的数据,原型是size_t fwrite(const void *ptr, size_t size, size_t n, FILE *stream);参数ptr是要写入的数据指针,void*的指针可用任何类型的指针来替换,如char*、int *等等来替换;size是每块的字节数;n是要写的块数,如果成功,返回实际写入的块数(不是字节数),本函数一般用于二进制模式打开的文件中。</P>& i3 Q+ m" c0 j2 D6 v" A+ k
<P>例:</P>
3 P6 Y' U4 y% [5 A( ]6 i<P> char x[]="I Love You";
) ^) V! m( B. J7 @" S fwire(x, 6,12,fp);//写入6*12=72字节</P>
0 X& y- h5 Q! F; ^' \1 \: k<P> 将把"I Love"写到流fp中12次,共72字节</P>
# R+ \- q* I7 a; ?" z5 @<P>15.tmpfile()8 |, O3 K, x! K- o% R! x( w% h
其原型是FILE *tmpfile(void); 生成一个临时文件,以"w+b"的模式打开,并返回这个临时流的指针,如果失败返回NULL。在程序结束时,这个文件会被自动删除。</P>9 a& W0 u' H* u0 s( H4 v% `
<P>例:FILE *fp=tmpfile();</P>
S$ x [6 |4 A5 c' C+ T4 H<P>16.tmpnam();
2 h) u2 Y7 b+ b" Q2 R) t1 K; \ 其原型为char *tmpnam(char *s); 生成一个唯一的文件名,其实tmpfile()就调用了此函数,参数s用来保存得到的文件名,并返回这个指针,如果失败,返回NULL。</P>6 E' R F, c# b
5 x1 M' ^$ F6 Y) ~( b1 E. I<P>1 c/ h7 l3 o% q1 G4 T0 I
</P> |
zan
|