数学建模社区-数学中国
标题:
borland c++ bulder的文件操作总结-1
[打印本页]
作者:
韩冰
时间:
2005-1-26 13:07
标题:
borland c++ bulder的文件操作总结-1
<
>、基于C的文件操作
) H5 }; e$ P) M5 G1 d) N
在ANSI C中,对文件的操作分为两种方式,即流式文件操作和I/O文件操作,下面就分别介绍之。</P>
' i, u5 ?, Z% H8 Z
<
>一、流式文件操作
' o3 Q. K0 r. e2 X) u: ?
这种方式的文件操作有一个重要的结构FILE,FILE在stdio.h中定义如下:</P>
% P3 J, |3 x6 n1 l* r
<
>typedef struct {
8 d0 j+ q( u, d2 K; r
int level; /* fill/empty level of buffer */
2 A- Q* g: @1 r" u& p) s& x% N1 a
unsigned flags; /* File status flags */
7 s. r1 P4 o5 Q# Y* _" E
char fd; /* File descriptor */
" L5 _4 M& @$ {; L* c2 _
unsigned char hold; /* Ungetc char if no buffer */
+ K6 s( ?# C7 G+ H+ M
int bsize; /* Buffer size */
5 s4 g! P4 D1 [7 W/ C
unsigned char _FAR *buffer; /* Data transfer buffer */
* X1 w) e% c$ {% w, O6 T4 k
unsigned char _FAR *curp; /* Current active pointer */
0 i. h4 l6 A5 Y4 @) q
unsigned istemp; /* Temporary file indicator */
O3 j1 Y) ?& `/ i& {1 F" w
short token; /* Used for validity checking */
: }* D4 I+ _' A
} FILE; /* This is the FILE object */</P>
/ Q! g7 M7 d3 }7 i
<
> FILE这个结构包含了文件操作的基本属性,对文件的操作都要通过这个结构的指针来进行,此种文件操作常用的函数见下表 函数 功能
9 K9 e: Q1 a- i- D
fopen() 打开流
0 |8 F- W, k& Q8 c) B9 j, Q
fclose() 关闭流
9 {2 F2 A- J/ B" S8 f' v* Z
fputc() 写一个字符到流中
1 g5 B) u5 X) n2 w# I
fgetc() 从流中读一个字符
+ A3 ~& A+ x7 f& Z! J# N
fseek() 在流中定位到指定的字符
0 R0 F* `0 e% D7 s7 R
fputs() 写字符串到流
2 h# C* y" L2 \" W6 p
fgets() 从流中读一行或指定个字符
3 V5 D3 w7 C4 V9 Z) o0 A% w: R- @. u7 v
fprintf() 按格式输出到流
$ D3 }- K1 \$ |( y$ [" `' U
fscanf() 从流中按格式读取
* {& h. |/ _7 R3 y
feof() 到达文件尾时返回真值
! X3 l, k Q4 U' x {% r
ferror() 发生错误时返回其值
0 |& W2 }& X- H# Y& d
rewind() 复位文件定位器到文件开始处
6 Y* f0 X9 n0 l' {! H( ]+ B
remove() 删除文件
0 l: s, F6 j1 g9 W
fread() 从流中读指定个数的字符
, z! `! b9 `% d5 k: c
fwrite() 向流中写指定个数的字符
2 E% L0 [, ?1 n, |7 ^* h$ k& {
tmpfile() 生成一个临时文件流
, p5 [$ E$ L% k, _" L. R
tmpnam() 生成一个唯一的文件名 </P>
" N& P. X6 o; s. @2 Z7 w
<
>
: H5 k- k* a1 R! ?# i1 J8 B
下面就介绍一下这些函数</P>
( P8 [3 V! |' k' |
<
>1.fopen()
9 H( B$ T5 h' E6 P5 ?( ^) y
fopen的原型是:FILE *fopen(const char *filename,const char *mode),fopen实现三个功能</P>
* o1 O7 M+ z+ w8 H& k) ^0 O
<
>为使用而打开一个流
) r( q& h2 e/ G6 R/ B3 Z u. n
把一个文件和此流相连接
) p. Z3 K0 R! k
给此流返回一个FILR指针
s P3 d+ e! |8 _* F4 {: h
参数filename指向要打开的文件名,mode表示打开状态的字符串,其取值如下表</P>
) C( o- ^( a- E/ b9 T6 A0 {+ |
<
>字符串 含义
7 K; ^; y) X' V
"r" 以只读方式打开文件
7 ^3 y E5 F' Z2 Y
"w" 以只写方式打开文件
# i6 X* Y* h0 x- X+ b- {8 y6 d3 I
"a" 以追加方式打开文件
5 m( z) T1 J8 d, S
"r+" 以读/写方式打开文件,如无文件出错
" B% w% B( T! P9 n8 T- q9 V, i
"w+" 以读/写方式打开文件,如无文件生成新文件 </P>
8 s# u! J7 y( i$ M% D+ R
<
> 一个文件可以以文本模式或二进制模式打开,这两种的区别是:在文本模式中回车被当成一个字符'\n',而二进制模式认为它是两个字符0x0D,0x0A;如果在文件中读到0x1B,文本模式会认为这是文件结束符,也就是二进制模型不会对文件进行处理,而文本方式会按一定的方式对数据作相应的转换。</P>
3 d7 K( L* q) @! I
<
> 系统默认的是以文本模式打开,可以修改全部变量_fmode的值来修改这个设置,例如_fmode=O_TEXT;就设置默认打开方式为文本模式;而_fmode=O_BINARY;则设置默认打开方式是二进制模式。</P>
7 b& z' E% \$ X/ Z$ m' B5 l
<
> 我们也可以在模式字符串中指定打开的模式,如"rb"表示以二进制模式打开只读文件,"w+t"或"wt+"表示以文本模式打开读/写文件。</P>
- [* ?1 M2 F7 \8 C
<
> 此函数返回一个FILE指针,所以申明一个FILE指针后不用初始化,而是用fopen()来返回一个指针并与一个特定的文件相连,如果成败,返回NULL。</P>
9 X- e# r( }- F _$ N0 I/ o: W
<
>例:</P>
; h: C$ V$ g0 Z. g
<
> FILE *fp;
: D( X( |. K5 Y n. m4 @. A! n
if(fp=fopen("123.456","wb"))
. U: D/ _5 Z" z5 U' D4 g# L1 w
puts("打开文件成功");
) a+ d [8 h+ |4 C7 \
else
" t3 [/ P! o* A
puts("打开文件成败"); </P>
5 ^/ t0 O2 P% i8 z
<
>2.fclose()
+ ?2 }' q. ~1 B9 Q% T9 l
fclose()的功能就是关闭用fopen()打开的文件,其原型是:int fclose(FILE *fp);如果成功,返回0,失败返回EOF。</P>
: [. D% H) g4 o1 e9 K( n6 P
<
> 在程序结束时一定要记得关闭打开的文件,不然可能会造成数据丢失的情况,我以前就经常犯这样的毛病。</P>
1 J- g J" c$ H2 |& B
<
>例:fclose(fp);</P>
J! n) P! E$ k
<
>3.fputc()
4 ^2 ]4 B* E$ n; X; i' V
向流写一个字符,原型是int fputc(int c, FILE *stream); 成功返回这个字符,失败返回EOF。</P>
' L2 s% `; u+ p9 o9 `
<
>例:fputc('X',fp);</P>
# |$ O% Y+ U. @ [ h( ~
<
>4.fgetc()
/ w3 D8 x8 i2 p5 f
从流中读一个字符,原型是int fputc(FILE *stream); 成功返回这个字符,失败返回EOF。</P>
& h4 H3 W6 S i( m! ` b
<
>例:char ch1=fgetc(fp);</P>
/ U4 G( a! G" z; w
<
>5. fseek()
# v' u: G9 v u$ q
此函数一般用于二进制模式打开的文件中,功能是定位到流中指定的位置,原型是int fseek(FILE *stream, long offset, int whence);如果成功返回0,参数offset是移动的字符数,whence是移动的基准,取值是</P>
6 z1 J. C* q5 B4 b
<
>符号常量 值 基准位置
3 I/ E# f4 ~# N# y, S
SEEK_SET 0 文件开头
; t8 j+ V& R7 a8 M7 z8 T4 x
SEEK_CUR 1 当前读写的位置
+ ~. a7 \3 ?7 L6 Y# f B) V
SEEK_END 2 文件尾部 </P>
. g3 ~4 Y/ n) ]4 |1 G
<
>例:fseek(fp,1234L,SEEK_CUR);//把读写位置从当前位置向后移动1234字节(L后缀表示长整数)</P>
7 @$ z' c' V9 `$ @
<
> fseek(fp,0L,2);//把读写位置移动到文件尾</P>
, A; V* o5 _* m4 [
<
>6.fputs()
0 I! E. ?( H; _1 C
写一个字符串到流中,原型int fputs(const char *s, FILE *stream); </P>
) R1 {7 |: n* T6 c% u
<
>例:fputs("I Love You",fp);</P>
2 @$ g* s+ {) Z% L
<
>7.fgets()
% M) F. w8 _0 f: i9 q# w7 |6 @5 u8 _' u
从流中读一行或指定个字符,原型是char *fgets(char *s, int n, FILE *stream); 从流中读取n-1个字符,除非读完一行,参数s是来接收字符串,如果成功则返回s的指针,否则返回NULL。</P>
: ^/ r* L# \8 Y& o; D; @/ S
<
>例:如果一个文件的当前位置的文本如下</P>
, q) n L4 C! e& t0 H8 X/ }
<
>Love ,I Have</P>
" S3 D8 q' \+ ~2 F/ `6 o' z
<P>But ........</P>
- a0 y1 b" |* u; d
<P>如果用</P>
8 s- W! d0 P) @: F9 P5 Q- C, Y
<P> fgets(str1,4,file1);</P>
& N) N+ \7 _! K% Y/ ?' A
<P>则执行后str1="Lov",读取了4-1=3个字符,而如果用</P>
0 S- f+ t' K1 _- i
<P> fgets(str1,23,file1);</P>
% L1 J' r f9 T1 N( l& q( f& j
<P>则执行str="Love ,I Have",读取了一行(不包括行尾的'\n')。</P>
) J2 w& f9 x6 |. e
<P>8.fprintf()
9 U+ O$ R& m) E
按格式输入到流,其原型是int fprintf(FILE *stream, const char *format[, argument, ...]);其用法和printf()相同,不过不是写到控制台,而是写到流罢了</P>
" U6 x5 o# `# W! L) o6 @
<P>例:fprintf(fp,"%2d%s",4,"Hahaha");</P>
8 w5 ~7 h9 U- m. ~7 M4 P
<P>9.fscanf()
1 H/ L# ~ V& M9 h
从流中按格式读取,其原型是int fscanf(FILE *stream, const char *format[, address, ...]);其用法和scanf()相同,不过不是从控制台读取,而是从流读取罢了。</P>
; D! A. e# ?$ T6 {% I: h
<P>例:fscanf(fp,"%d%d" ,&x,&y);</P>
9 ?" |* Z1 G# ?# D5 N
<P>10.feof()
G$ } ~* Z. D& ?
检测是否已到文件尾,是返回真,否则返回0,其原型是int feof(FILE *stream);</P>
& r8 t; v. Z q! d
<P>例:if(feof(fp))printf("已到文件尾");</P>
3 T9 N7 e4 ~( Y/ c
<P>11.ferror()
% T- U+ Z' O- o4 G! ~
原型是int ferror(FILE *stream);返回流最近的错误代码,可用clearerr()来清除它,clearerr()的原型是void clearerr(FILE *stream);</P>
8 T% V( B5 M8 l! G- b/ F% ^) j
<P>例:printf("%d",ferror(fp));</P>
& d, z+ x4 H/ H/ ?9 a# k) }
<P>12.rewind()
1 V n I6 P& Q% V# m
把当前的读写位置回到文件开始,原型是void rewind(FILE *stream);其实本函数相当于fseek(fp,0L,SEEK_SET);</P>
5 k/ H% _- W3 O& p* ]
<P>例:rewind(fp);</P>
* Z; t6 H9 J; ]3 [( X
<P>12.remove()
; C {( {! c, m0 x2 ]0 d/ @
删除文件,原型是int remove(const char *filename); 参数就是要删除的文件名,成功返回0。</P>
" O6 L2 E% J3 f1 E) B1 I" |" k
<P>例:remove("c:\\io.sys");</P>
. j$ [1 d2 @2 E. i. x
<P>13.fread()
4 \3 Z! ~) V1 Z; [$ K% C
从流中读指定个数的字符,原型是size_t fread(void *ptr, size_t size, size_t n, FILE *stream);参数ptr是保存读取的数据,void*的指针可用任何类型的指针来替换,如char*、int *等等来替换;size是每块的字节数;n是读取的块数,如果成功,返回实际读取的块数(不是字节数),本函数一般用于二进制模式打开的文件中。</P>
& ^2 a% C+ h5 V Y
<P>例:</P>
* H% V. @+ D! b+ |- h) C, e0 {
<P> char x[4230];
, S' j' Z+ \1 |) U& G) Y
FILE *file1=fopen("c:\\msdos.sys","r");
( z3 V7 _! W+ w; F3 M ]
fread(x,200,12 ,file1);//共读取200*12=2400个字节</P>
' {, X5 U/ u& B
<P>14.fwrite()
) w6 y. j+ |* u0 l2 x% n! I
与fread对应,向流中写指定的数据,原型是size_t fwrite(const void *ptr, size_t size, size_t n, FILE *stream);参数ptr是要写入的数据指针,void*的指针可用任何类型的指针来替换,如char*、int *等等来替换;size是每块的字节数;n是要写的块数,如果成功,返回实际写入的块数(不是字节数),本函数一般用于二进制模式打开的文件中。</P>
" g: W2 a' z6 J, [4 f
<P>例:</P>
3 [6 R1 p9 O9 R/ ^& `5 }2 v8 I( U
<P> char x[]="I Love You";
$ K3 p5 i: b/ {6 v2 I
fwire(x, 6,12,fp);//写入6*12=72字节</P>
& k( Z" W: H* |# W4 i
<P> 将把"I Love"写到流fp中12次,共72字节</P>
. h4 u) Y/ \0 l S' M4 O! s# a
<P>15.tmpfile()
+ _0 h# H* q& G, }0 `
其原型是FILE *tmpfile(void); 生成一个临时文件,以"w+b"的模式打开,并返回这个临时流的指针,如果失败返回NULL。在程序结束时,这个文件会被自动删除。</P>
9 E6 u* z* [$ l1 o3 W* I
<P>例:FILE *fp=tmpfile();</P>
6 b: m! E$ T% h1 ~8 C' E
<P>16.tmpnam();
8 K: U8 K( b. i' |, ]. i- R2 `: T
其原型为char *tmpnam(char *s); 生成一个唯一的文件名,其实tmpfile()就调用了此函数,参数s用来保存得到的文件名,并返回这个指针,如果失败,返回NULL。</P>
( }& Q3 ]* U/ L* L/ M
! e$ L! x0 G; K6 x- t: S
<P>
0 _& L4 p4 ?4 \0 k. p; ^
</P>
欢迎光临 数学建模社区-数学中国 (http://www.madio.net/)
Powered by Discuz! X2.5