|
请使用QQ关联注册PLM之家,学习更多关于内容,更多精彩原创视频供你学习!
您需要 登录 才可以下载或查看,没有账号?注册
x
ofstream是从内存到硬盘,ifstream是从硬盘到内存,其实所谓的流缓冲就是内存空间;
+ |9 Z8 ]; p( M( e1 k/ a
& `9 H/ [( K K1 A# r* a% o 在C++中,有一个stream这个类,所有的I/O都以这个“流”类为基础的,包括我们要认识的文件I/O,stream这个类有两个重要的运算符:
' d( D9 j* r+ Q$ c/ q
/ b7 Q! H- x+ [6 B2 G# ^ E 1、插入器(<<)
4 d) u- d- q( }, {0 n! r* }4 D % n- [5 c; u% @0 S+ D! y
向流输出数据。比如说系统有一个默认的标准输出流(cout),一般情况下就是指的显示器,所以,cout<<"Write Stdout"<<'\n';就表示把字符串"Write Stdout"和换行字符('\n')输出到标准输出流。: @; |# U7 {( \& D. @( V3 b
4 a8 q- ^# u: C' U6 d* [+ y 2、析取器(>>)
- j6 L$ {9 p) a, u2 l. h2 @3 S - w I% k% T' U$ V$ v$ S
从流中输入数据。比如说系统有一个默认的标准输入流(cin),一般情况下就是指的键盘,所以,cin>>x;就表示从标准输入流中读取一个指定类型(即变量x的类型)的数据。
& @( I' ]: d7 B5 R. @/ v9 [ , {( d: J9 a' L/ ~3 r: P2 `9 z
在C++中,对文件的操作是通过stream的子类fstream(file stream)来实现的,所以,要用这种方式操作文件,就必须加入头文件fstream.h。下面就把此类的文件操作过程一一道来。0 y9 { U+ A0 I0 k
z& b$ h6 D1 y2 A5 C( a9 v; u
一、打开文件
; R- j! V6 \) b# j) N: v3 } 9 `5 G7 n- v2 a. A% k
在fstream类中,有一个成员函数open(),就是用来打开文件的,其原型是:& U6 u" A& J% ?( b" @2 \; z/ ]
/ H# u0 Z4 g0 t$ q void open(const char* filename,int mode,int access);参数:1 f5 Z# ?6 P4 a/ x: t$ h
4 C: s* z" W3 j1 a2 k+ H f
filename: 要打开的文件名
% @0 | r8 `$ i& c. d
" |! O8 e: G# _$ X0 B6 [ mode: 要打开文件的方式1 V$ t2 f& Y. g6 ?2 L# G
3 a6 D/ W& i+ \+ D8 S) M access: 打开文件的属性, O+ `" [/ H3 W) b
' x% M! ~' _5 V7 w 打开文件的方式在类ios(是所有流式I/O类的基类)中定义,常用的值如下:
6 ~6 H/ h q4 `% Z
0 N/ R. [; K3 O4 w ios::app: 以追加的方式打开文件, B; X' |( c2 z0 J. X' t
- A' z( C' W o" o1 O4 e- u ios::ate: 文件打开后定位到文件尾,ios:app就包含有此属性0 w* c% E7 J2 ]6 H. C* k
* |! N% b& F: y3 A" c
ios::binary: 以二进制方式打开文件,缺省的方式是文本方式。两种方式的区别见前文- [% j. ]. h! M& k, R
j0 z. I! ?1 S, \) U& v3 s ios::in: 文件以输入方式打开(文件数据输入到内存)
2 b4 U/ q9 }! o+ j
6 Q# I! f) U8 x" O ios::out: 文件以输出方式打开(内存数据输出到文件)
# n% V# u- g7 g! n: } & @/ D; p) {0 @5 ~0 i& g% q
ios::nocreate: 不建立文件,所以文件不存在时打开失败4 \4 \3 G$ G1 {7 ?& r0 f4 v. b6 l
$ s. e# G) u, K1 C1 y3 p ios::noreplace:不覆盖文件,所以打开文件时如果文件存在失败% m/ J/ v$ P! j
' Z5 }" a7 P) d
ios::trunc: 如果文件存在,把文件长度设为07 O" C8 m2 ^; ^ F6 ?! r; J, W$ h0 s
6 [, s) {" ^) l2 h0 z 可以用“或”把以上属性连接起来,如ios::out|ios::binary" W" ?/ H9 o9 C
, m- K3 b' t) l7 x4 a% v1 n 打开文件的属性取值是:
$ n) [% J% m, |( N6 t; H' ?
" F# o6 e5 x. O3 t 0:普通文件,打开访问9 ^* n9 h1 ~" X# y$ P
4 J/ x# Q0 w0 R5 c
1:只读文件
# M0 d* U, U# |- R# A- ?* u# N# f D
. A; S& _: A6 E' x5 D) V 2:隐含文件1 T$ R9 h' ~1 a6 a5 E- a' f" R
3 e* S1 y$ w* u# i m* r T, a
4:系统文件
% e# i9 F; V1 K( Q
3 x6 `/ B7 v" Q$ f* I# K% J; R 可以用“或”或者“+”把以上属性连接起来,如3或1|2就是以只读和隐含属性打开文件。
/ [8 G0 E" R- a0 O0 |+ o5 r; @ t ; C" E7 h3 B8 B" V3 q" H4 d2 C
例如:以二进制输入方式打开文件c:\config.sys
0 ]: A2 Z$ e) k5 k. J# b - Q8 \" w6 Y7 Z) C* u+ A. G
fstream file1;
; V9 _, y+ a! w' t ; C/ I: x( Q$ B; F
file1.open("c:\\config.sys",ios::binary|ios::in,0);# O |# D0 N8 c. R
2 A! Y1 D2 x8 x5 Y e 如果open函数只有文件名一个参数,则是以读/写普通文件打开,即:
# |5 r/ Z! X5 F8 m & A* _5 E2 ~5 h! H) R3 u+ S" d3 J
file1.open("c:\\config.sys"); <=> file1.open("c:\\config.sys",ios::in|ios::out,0);
+ C9 L: N$ S+ `: w- @
( L& F2 T2 h |& f 另外,fstream还有和open()一样的构造函数,对于上例,在定义的时侯就可以打开文件了:& C; G7 o: ~2 S& G, \" [* S
# K$ U$ U, k+ o$ P% n K fstream file1("c:\\config.sys"); 特别提出的是,fstream有两个子类:ifstream(input file stream)和ofstream(outpu file stream),ifstream默认以输入方式打开文件,而ofstream默认以输出方式打开文件。
$ B K- X |, L; E0 Y
0 c2 I M2 w; O5 l7 P6 V5 ]- p ifstream file2("c:\\pdos.def");//以输入方式打开文件1 l# m5 v0 w5 l8 F
. u# ?7 A# K/ G% W4 w3 A ofstream file3("c:\\x.123");//以输出方式打开文件 所以,在实际应用中,根据需要的不同,选择不同的类来定义:如果想以输入方式打开,就用ifstream来定义;如果想以输出方式打开,就用ofstream来定义;如果想以输入/输出方式来打开,就用fstream来定义。
' u: Y: `$ u, ^! k ! Z4 x8 ~: V0 ]
二、关闭文件1 Z: @% O, q8 n, p# c( T, m; |
" Y" v) |+ o6 a5 u$ Y, r' @ Y
打开的文件使用完成后一定要关闭,fstream提供了成员函数close()来完成此操作,如:file1.close();就把file1相连的文件关闭。
: H4 {' x- H; G& p! T
5 O- x5 j; k; b d! ^ 三、读写文件
1 z! g! j q. y9 R( t# C5 M, q/ j. e4 W
2 N7 i T0 ], p0 P" h# J4 ` 读写文件分为文本文件和二进制文件的读取,对于文本文件的读取比较简单,用插入器和析取器就可以了;而对于二进制的读取就要复杂些,下要就详细的介绍这两种方式0 p! ~% }- H" X" [+ X6 p' Z
|- m$ D/ ~/ `% b, c# G% K
1、文本文件的读写2 r u7 d. b! p& _1 Y( P/ f$ ?* L
, N# T) ?6 V% W( F4 a 文本文件的读写很简单:用插入器(<<)向文件输出;用析取器(>>)从文件输入。假设file1是以输入方式打开,file2以输出打开。示例如下:, \/ j! U) g. k& u, [
5 M& W t" @2 S file2<<"I Love You";//向文件写入字符串"I Love You") G B7 w) ?1 }( u
6 Y$ s; E5 Q8 s, v9 a( e int i;
+ l6 x, } D, H" x
( `/ P: p' ^) x9 a file1>>i;//从文件输入一个整数值。" C; {7 E; E6 V7 S8 h
0 q5 V( M! S6 G/ w/ \ 这种方式还有一种简单的格式化能力,比如可以指定输出为16进制等等,具体的格式有以下一些/ y/ F6 c! a" l- P6 z/ F! ?
& J' y3 X/ n/ ^/ { 操纵符 功能 输入/输出
* [1 V" l7 n- I! `2 k+ k- a+ I 0 v# R% ]5 {4 f9 q+ ]/ v( [
dec 格式化为十进制数值数据 输入和输出
# c( F% x" j+ l$ N$ y3 Z8 L+ y ' V. |6 U& {7 g0 v) [
endl 输出一个换行符并刷新此流 输出
1 z" q" T" `) ~$ ]) l; f 8 q# W7 X) W" v; P9 u* D0 X
ends 输出一个空字符 输出6 L% ^9 v' r: s5 m7 |
; Q& N3 E/ e* k' y2 i hex 格式化为十六进制数值数据 输入和输出
% G1 t6 d) O% J2 V: ]1 R- h2 o' A- K
4 I1 N5 O: g$ A! W7 v: w- |' i" s0 k oct 格式化为八进制数值数据 输入和输出
% g5 E* @# d6 K& X, x7 X O
& x) {* k' }. R& Y. P) b setpxecision(int p) 设置浮点数的精度位数 输出
* C9 b. n& X: J5 E% U
$ C% S) a; b( l2 L 比如要把123当作十六进制输出:file1<
/ o7 z/ k$ b4 r; h: l- i6 u 2、二进制文件的读写
1 _5 u5 L- `1 W
/ e/ E4 L4 ^! J3 f9 a ①put()
8 E1 u# K5 ^, N) x$ f W/ O$ T2 G7 @/ n
put()函数向流写入一个字符,其原型是ofstream &put(char ch),使用也比较简单,如file1.put('c');就是向流写一个字符'c'。
; v+ C) x3 Q8 m$ W( O
* @! l1 U! h+ C8 U2 P# _ ②get(); b2 Q" M) j+ E6 P3 U) G) {
: J& Q) ^" e! d }$ { Y get()函数比较灵活,有3种常用的重载形式:5 v' V; Y8 k- c, C9 ~, B
: s/ c0 E3 v! E# i
一种就是和put()对应的形式:ifstream &get(char &ch);功能是从流中读取一个字符,结果保存在引用ch中,如果到文件尾,返回空字符。如file2.get(x);表示从文件中读取一个字符,并把读取的字符保存在x中。
1 C+ a @# w$ O3 X% `) K. n 0 O) r$ u7 V, V% @7 \9 _4 X
另一种重载形式的原型是: int get();这种形式是从流中返回一个字符,如果到达文件尾,返回EOF,如x=file2.get();和上例功能是一样的。
* t* V: Z- Q- ~* m " K4 ]$ ^& H6 g/ g$ r! E( C |& K+ }
还有一种形式的原型是:ifstream &get(char *buf,int num,char delim='\n');这种形式把字符读入由 buf 指向的数组,直到读入了 num 个字符或遇到了由 delim 指定的字符,如果没使用 delim 这个参数,将使用缺省值换行符'\n'。例如:0 ]; A6 T: h+ q8 N7 y
D) D6 T3 M, k- H: }& I file2.get(str1,127,'A'); //从文件中读取字符到字符串str1,当遇到字符'A'或读取了127个字符时终止。
# |; l' w% Z) v$ E0 c, F" K+ v ; m* V& f. C/ S4 `% m
③读写数据块1 H& E+ P. N- g; u
3 p+ S: K- {$ M5 ~1 h& L; _ 要读写二进制数据块,使用成员函数read()和write()成员函数,它们原型如下:, J: U3 I+ _$ b i
u) n" H% W3 b3 N3 y! \" O' \- f
read(unsigned char *buf,int num);
- T1 l H8 ^' _ R+ o
1 i3 |* m. \$ n: z( f+ ^9 q, G write(const unsigned char *buf,int num);) w1 o1 l" G8 M, x! k/ p( g
% t* {) M& G8 z- h1 g8 r1 m O
read()从文件中读取 num 个字符到 buf 指向的缓存中,如果在还未读入 num 个字符时就到了文件尾,可以用成员函数 int gcount();来取得实际读取的字符数;而 write() 从buf 指向的缓存写 num 个字符到文件中,值得注意的是缓存的类型是 unsigned char *,有时可能需要类型转换。8 ~( _( {* ]$ q {4 U
+ f6 n2 W# N% r6 V5 J
例:
! F ?! E4 n/ K* I 8 h( O0 u$ {6 Q0 K
unsigned char str1[]="I Love You";5 E" u2 S- d" k+ S2 E _8 i
- p& k; V0 d$ l ?4 b* E. a9 _$ T
int n[5];" z: H _, I9 H- u
2 V. S/ b# x8 {) W% @* Z9 w P
ifstream in("xxx.xxx");& @. W H) E/ w
q7 h% M: B L4 u% S5 h ofstream out("yyy.yyy");
( `6 m/ Q& m5 z 8 t. R% Q! u( ]- W
out.write(str1,strlen(str1));//把字符串str1全部写到yyy.yyy中' P5 O) @8 B1 H D
6 ~& m% B3 e5 u$ m6 V3 R in.read((unsigned char*)n,sizeof(n));//从xxx.xxx中读取指定个整数,注意类型转换
6 Q' N$ s% S0 x8 Q6 {3 h. c& w
u7 J2 o* |$ j/ L8 l) A. X in.close();out.close(); 四、检测EOF$ C+ [2 H# ]0 S% q1 ~ \- K
: f! p, v) L9 h6 Q- Q5 A! c9 H
成员函数eof()用来检测是否到达文件尾,如果到达文件尾返回非0值,否则返回0。原型是int eof();
' b* f9 ^2 r( D, F4 K+ i, S% { 4 B% T* Y) `7 w5 c1 A6 A( m5 E
例: if(in.eof()) ShowMessage("已经到达文件尾!");
/ H Q5 C! `* ~8 D. m2 T & X) I; p8 i* u
五、文件定位* R2 q5 Q: D' _! V% J/ @- |
: k+ c2 l$ D* o4 z 和C的文件操作方式不同的是,C++ I/O系统管理两个与一个文件相联系的指针。一个是读指针,它说明输入操作在文件中的位置;另一个是写指针,它下次写操作的位置。每次执行输入或输出时,相应的指针自动变化。所以,C++的文件定位分为读位置和写位置的定位,对应的成员函数是seekg()和seekp()。seekg()是设置读位置, seekp是设置写位置。它们最通用的形式如下:
/ i: T* G0 \. J& s: x ! W& b3 N% m2 a# t/ L0 S. D- N
istream &seekg(streamoff offset,seek_dir origin);$ p4 o& Q0 f6 D+ R& U
) z* Q" J7 ?5 \1 e
ostream &seekp(streamoff offset,seek_dir origin);
L R& a* }6 M1 J 2 b: n* u' Z2 J
streamoff定义于 iostream.h 中,定义有偏移量 offset 所能取得的最大值,seek_dir 表示移动的基准位置,是一个有以下值的枚举:4 g* }4 t0 _* [3 W ]/ ~
0 @5 q2 X% E( D9 p9 v% Q) t: m
ios::beg: 文件开头 y4 U( z# P( s: w& ?: x
: m8 M2 v5 P ]) A& u0 p' o, {
ios::cur: 文件当前位置
8 P! I( d' k7 d9 g4 Q4 X 1 e) q! q( {6 u& Z' {: C! k
ios::end: 文件结尾7 }! R% u6 g+ @6 J
. o3 Y- [" A+ F* _: [% U5 U* C 这两个函数一般用于二进制文件,因为文本文件会因为系统对字符的解释而可能与预想的值不同。例:
( |- e1 x; y# r" w& R. k2 i
3 a" |/ U$ W9 X F. h. X file1.seekg(1234,ios::cur); //把文件的读指针从当前位置向后移1234个字节
1 ?/ K1 H0 Z1 o( S" C, w! }
4 g1 K2 |$ w5 Y2 J, P file2.seekp(1234,ios::beg); //把文件的写指针从文件开头向后移1234个字节
9 b, ]2 a/ W. c. z
; o" W) H1 n5 V6 z3 k& d! L; @ |
|