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

我的地盘我做主
该用户从未签到
 |
< >、基于C的文件操作 5 F0 C; w1 @ ^
在ANSI C中,对文件的操作分为两种方式,即流式文件操作和I/O文件操作,下面就分别介绍之。</P>: X/ e) D& x% T8 k# M9 M) N
< >一、流式文件操作1 P+ g9 D$ B! d( z2 @# R
这种方式的文件操作有一个重要的结构FILE,FILE在stdio.h中定义如下:</P>
5 }0 t! w1 [) c: C t< >typedef struct {
: l$ r$ ~( C6 f0 F/ r8 r. Xint level; /* fill/empty level of buffer */
5 Y- W. B, B2 G* @0 l- h3 f% hunsigned flags; /* File status flags */( @1 c0 l' _, e& Q8 G \; @
char fd; /* File descriptor */
# ?* t3 r9 o% ^6 U2 }: N, S1 t" H6 wunsigned char hold; /* Ungetc char if no buffer */
% z7 ~2 J4 m% Y0 n6 g5 b- F, yint bsize; /* Buffer size */
/ }) r$ R; E+ p) X: T% y6 ?unsigned char _FAR *buffer; /* Data transfer buffer */
2 z4 a$ l7 J0 Bunsigned char _FAR *curp; /* Current active pointer */' b) X( s; v5 i0 ?4 I
unsigned istemp; /* Temporary file indicator *// p$ G& @/ k- s; v
short token; /* Used for validity checking */) S* L0 y+ ^3 h/ `# f
} FILE; /* This is the FILE object */</P>
$ i: I4 G6 W' I: \< > FILE这个结构包含了文件操作的基本属性,对文件的操作都要通过这个结构的指针来进行,此种文件操作常用的函数见下表 函数 功能 ! g8 P1 A" n, _* d7 W( X
fopen() 打开流 4 h$ }$ H" o8 R/ C+ R! m
fclose() 关闭流 1 F- @# ~3 X7 X: E {& d
fputc() 写一个字符到流中 4 ]/ k/ ?% u; P
fgetc() 从流中读一个字符
4 V) R" T/ F. \fseek() 在流中定位到指定的字符 * ~/ L8 F9 s2 ?2 ^0 e0 g
fputs() 写字符串到流 8 C* n& F# a! Q3 S/ G
fgets() 从流中读一行或指定个字符 " e. M6 f4 h+ m2 F" T7 u) c0 T
fprintf() 按格式输出到流 % h& a0 t6 a6 ?& Q& t
fscanf() 从流中按格式读取 $ s2 a0 N6 d9 J: N6 Y; Z
feof() 到达文件尾时返回真值 : h$ D9 x* ]+ E4 ~- w
ferror() 发生错误时返回其值
" t$ l: |" W U( J: L& zrewind() 复位文件定位器到文件开始处
/ B3 R: ~% G3 z% H, [9 i& w9 premove() 删除文件
5 p4 m& G0 h' |+ @fread() 从流中读指定个数的字符 ' J# q+ k6 V8 l& v$ u& |. c F2 w
fwrite() 向流中写指定个数的字符 . v V4 s) ^) y
tmpfile() 生成一个临时文件流 ! a( _4 T1 p# f
tmpnam() 生成一个唯一的文件名 </P>3 t" d8 d6 {! L/ x
< >
& v$ X2 g6 C( j+ O 下面就介绍一下这些函数</P>& n. N# P: e! Y8 } v6 I- m
< >1.fopen()
9 l5 w7 U' s( k6 M4 M fopen的原型是:FILE *fopen(const char *filename,const char *mode),fopen实现三个功能</P>+ h. t" E9 Z7 P4 M9 c1 A
< >为使用而打开一个流
" C) j; u* r( A( u$ z# a把一个文件和此流相连接 ) Q& ~' U5 |0 ^& W: p0 Q+ @/ h# s
给此流返回一个FILR指针; {; ~8 R' k7 q. ]0 d3 K& @% W: B
参数filename指向要打开的文件名,mode表示打开状态的字符串,其取值如下表</P>
# Y$ b) _* x$ p6 |, y3 F3 |< >字符串 含义 * u3 Y4 _' q' S8 i8 c# E
"r" 以只读方式打开文件 " W: \7 q3 D% f. c4 R+ p5 `/ U% g
"w" 以只写方式打开文件
% N _& b0 c1 X0 b) Q) c"a" 以追加方式打开文件
& }! J. c5 }/ n% B( z3 l"r+" 以读/写方式打开文件,如无文件出错
- W9 b7 y; N" f0 k/ Y0 }$ E"w+" 以读/写方式打开文件,如无文件生成新文件 </P>5 F5 d( p# B& g
< > 一个文件可以以文本模式或二进制模式打开,这两种的区别是:在文本模式中回车被当成一个字符'\n',而二进制模式认为它是两个字符0x0D,0x0A;如果在文件中读到0x1B,文本模式会认为这是文件结束符,也就是二进制模型不会对文件进行处理,而文本方式会按一定的方式对数据作相应的转换。</P>
& v/ F6 c/ A Z- t0 m9 f< > 系统默认的是以文本模式打开,可以修改全部变量_fmode的值来修改这个设置,例如_fmode=O_TEXT;就设置默认打开方式为文本模式;而_fmode=O_BINARY;则设置默认打开方式是二进制模式。</P>
, n! M2 C: x, j- N# s< > 我们也可以在模式字符串中指定打开的模式,如"rb"表示以二进制模式打开只读文件,"w+t"或"wt+"表示以文本模式打开读/写文件。</P>4 Q1 l" }* |: U- V
< > 此函数返回一个FILE指针,所以申明一个FILE指针后不用初始化,而是用fopen()来返回一个指针并与一个特定的文件相连,如果成败,返回NULL。</P>' }' {" Z; d1 e# I' E5 d% F9 q, s! j
< >例:</P>
- \6 O8 [ v6 H% V# K" }< > FILE *fp; / z( z3 ~0 \; Z+ M2 \+ \
if(fp=fopen("123.456","wb"))
7 x% |7 A/ D* d8 n puts("打开文件成功");
- M6 ]' u, F( ]$ s6 Z else
. M. p5 B0 ?, W8 ` puts("打开文件成败"); </P>7 x: ^' {0 P2 v! H
< >2.fclose() N+ J; d* k$ M k6 H# Q& F9 S( m
fclose()的功能就是关闭用fopen()打开的文件,其原型是:int fclose(FILE *fp);如果成功,返回0,失败返回EOF。</P># p, J G4 Z9 s% z
< > 在程序结束时一定要记得关闭打开的文件,不然可能会造成数据丢失的情况,我以前就经常犯这样的毛病。</P>$ L3 F% h$ g( K0 b( |
< >例:fclose(fp);</P>
! d o& x; u5 e$ ~# G3 r7 R< >3.fputc()
% l3 S# |- {/ K. W5 c 向流写一个字符,原型是int fputc(int c, FILE *stream); 成功返回这个字符,失败返回EOF。</P>& U% Y7 }" k2 e: J; e+ Y: H
< >例:fputc('X',fp);</P>
( V! ]2 _; p2 v. L( I, o7 Y< >4.fgetc()
3 V" N5 v1 x& f3 i# n2 @8 } 从流中读一个字符,原型是int fputc(FILE *stream); 成功返回这个字符,失败返回EOF。</P>9 g0 f) O3 k; u) s4 a# ^
< >例:char ch1=fgetc(fp);</P>0 v c, V, k0 y2 Q2 Y5 d
< >5. fseek()
6 U; I% b" |: R9 p 此函数一般用于二进制模式打开的文件中,功能是定位到流中指定的位置,原型是int fseek(FILE *stream, long offset, int whence);如果成功返回0,参数offset是移动的字符数,whence是移动的基准,取值是</P>) i0 z2 W/ W( g. C2 Q
< >符号常量 值 基准位置 3 ~9 Z4 u! T7 x: [# b
SEEK_SET 0 文件开头 & {2 f9 H" J( Q# i
SEEK_CUR 1 当前读写的位置
+ ~5 U3 h2 d0 j0 f- fSEEK_END 2 文件尾部 </P>
1 A: E; p5 n2 h! S# Y5 ]< >例:fseek(fp,1234L,SEEK_CUR);//把读写位置从当前位置向后移动1234字节(L后缀表示长整数)</P>
: Z3 l7 J. N$ n- b& E# P) q, R< > fseek(fp,0L,2);//把读写位置移动到文件尾</P>. T, R+ t& E5 }9 f8 n* ]4 ?
< >6.fputs()
2 W) M- m8 m( F+ O 写一个字符串到流中,原型int fputs(const char *s, FILE *stream); </P>
; e- d u- t+ T; d, i. O< >例:fputs("I Love You",fp);</P>5 f2 [0 X" c3 m8 D4 P$ y; V' C
< >7.fgets()
2 P- C$ W4 r# _ g& E* W 从流中读一行或指定个字符,原型是char *fgets(char *s, int n, FILE *stream); 从流中读取n-1个字符,除非读完一行,参数s是来接收字符串,如果成功则返回s的指针,否则返回NULL。</P>6 T" J( s, j6 r) S
< >例:如果一个文件的当前位置的文本如下</P>
: M% z4 a0 P' h) u< >Love ,I Have</P>
5 T0 C9 `: D$ O& C, `<P>But ........</P>
: J. j6 K" W, a<P>如果用</P># D( |* N2 T( n
<P> fgets(str1,4,file1);</P># Q3 h n/ e* \3 Z7 x5 }% M
<P>则执行后str1="Lov",读取了4-1=3个字符,而如果用</P>
9 { d% R" n1 S7 O' h3 v<P> fgets(str1,23,file1);</P>
8 [' A4 p D1 P+ l) T9 R2 m/ f# f% z) }. B<P>则执行str="Love ,I Have",读取了一行(不包括行尾的'\n')。</P>7 G& d7 ^4 C. L/ N2 ^0 A3 S8 c
<P>8.fprintf()
& R+ j) \4 z5 g% Z! [- w6 h 按格式输入到流,其原型是int fprintf(FILE *stream, const char *format[, argument, ...]);其用法和printf()相同,不过不是写到控制台,而是写到流罢了</P>1 j$ D5 z' v9 z8 X* e7 T
<P>例:fprintf(fp,"%2d%s",4,"Hahaha");</P>
* P- f0 l0 K- ]2 L! R' X; A<P>9.fscanf(): ]& D9 R. d& A: `+ u
从流中按格式读取,其原型是int fscanf(FILE *stream, const char *format[, address, ...]);其用法和scanf()相同,不过不是从控制台读取,而是从流读取罢了。</P>0 i m: }5 [$ ?; A5 i
<P>例:fscanf(fp,"%d%d" ,&x,&y);</P>
( Z9 B; R7 e/ l! \: G) `<P>10.feof()
9 N0 v$ C: R/ D2 \6 R 检测是否已到文件尾,是返回真,否则返回0,其原型是int feof(FILE *stream);</P>( C+ ]8 `. d' q
<P>例:if(feof(fp))printf("已到文件尾");</P>
9 |' s+ G9 f8 w; t1 N<P>11.ferror()
6 J7 \/ f! p! B* w( {! n, E 原型是int ferror(FILE *stream);返回流最近的错误代码,可用clearerr()来清除它,clearerr()的原型是void clearerr(FILE *stream);</P>
( P% m5 N6 w6 v' P. P$ F7 B) G<P>例:printf("%d",ferror(fp));</P>% x/ N" Y8 X) G/ r4 }
<P>12.rewind()/ Z* U+ D0 Y. h ?" u% u
把当前的读写位置回到文件开始,原型是void rewind(FILE *stream);其实本函数相当于fseek(fp,0L,SEEK_SET);</P>
3 X, k: c7 Y9 q<P>例:rewind(fp);</P>
9 K4 W4 V6 e2 w @5 l0 h<P>12.remove()* J2 X: q2 ^% G% S4 X
删除文件,原型是int remove(const char *filename); 参数就是要删除的文件名,成功返回0。</P>( S: h! J" D# L. J" z
<P>例:remove("c:\\io.sys");</P>
* @. Y' O5 F" Q2 M<P>13.fread()
8 t+ d, r3 B* j, {* T& o0 w; b6 n 从流中读指定个数的字符,原型是size_t fread(void *ptr, size_t size, size_t n, FILE *stream);参数ptr是保存读取的数据,void*的指针可用任何类型的指针来替换,如char*、int *等等来替换;size是每块的字节数;n是读取的块数,如果成功,返回实际读取的块数(不是字节数),本函数一般用于二进制模式打开的文件中。</P>
2 i: ]! v+ J+ f$ h& }<P>例:</P>
: {6 H& s2 R+ \0 c; z<P> char x[4230];1 p% {1 }( s2 H9 A ~
FILE *file1=fopen("c:\\msdos.sys","r");$ X; j8 s2 B% R8 {- G
fread(x,200,12 ,file1);//共读取200*12=2400个字节</P>
* ?' Z& p/ D3 Z& k<P>14.fwrite()! u2 S/ A$ O* o
与fread对应,向流中写指定的数据,原型是size_t fwrite(const void *ptr, size_t size, size_t n, FILE *stream);参数ptr是要写入的数据指针,void*的指针可用任何类型的指针来替换,如char*、int *等等来替换;size是每块的字节数;n是要写的块数,如果成功,返回实际写入的块数(不是字节数),本函数一般用于二进制模式打开的文件中。</P>
6 N- p+ p/ v6 H; d% g3 S<P>例:</P>
7 `4 R. q: k* J2 \+ S<P> char x[]="I Love You";
+ A K. B& O/ L: V* r fwire(x, 6,12,fp);//写入6*12=72字节</P>3 [* ^9 t- e6 t9 B9 V4 A
<P> 将把"I Love"写到流fp中12次,共72字节</P>. ? ?; ~* s/ `2 U( P
<P>15.tmpfile()
% n q2 P( a- h/ M9 X 其原型是FILE *tmpfile(void); 生成一个临时文件,以"w+b"的模式打开,并返回这个临时流的指针,如果失败返回NULL。在程序结束时,这个文件会被自动删除。</P>
1 f4 v5 Y" L; H( Z% G/ d<P>例:FILE *fp=tmpfile();</P>: w% s6 j/ b: F7 f/ w
<P>16.tmpnam();
9 B a9 S$ C0 C4 g% e2 b4 r 其原型为char *tmpnam(char *s); 生成一个唯一的文件名,其实tmpfile()就调用了此函数,参数s用来保存得到的文件名,并返回这个指针,如果失败,返回NULL。</P>0 M" k; J+ p. `/ C/ T" P7 q* t+ J
9 R8 @' B5 z: Y+ D: W
<P>
( }$ H! S- k, h( J1 Y$ ^: o. s </P> |
zan
|