|
请使用QQ关联注册PLM之家,学习更多关于内容,更多精彩原创视频供你学习!
您需要 登录 才可以下载或查看,没有账号?注册
x
ofstream是从内存到硬盘,ifstream是从硬盘到内存,其实所谓的流缓冲就是内存空间;! Y8 V: z( l& O: E1 r
, Y- b" @8 _4 }( @. |( P; R c 在C++中,有一个stream这个类,所有的I/O都以这个“流”类为基础的,包括我们要认识的文件I/O,stream这个类有两个重要的运算符:4 N: Y& y$ z: N2 t
+ A% ^4 C' v9 ~1 C& j5 F% Z& O
1、插入器(<<)4 l* H- I8 c e, d8 |
2 M$ X1 ]5 i- L. {1 V# L 向流输出数据。比如说系统有一个默认的标准输出流(cout),一般情况下就是指的显示器,所以,cout<<"Write Stdout"<<'\n';就表示把字符串"Write Stdout"和换行字符('\n')输出到标准输出流。
0 ]0 i% h6 A9 o5 ]% ?
& d3 P7 k6 ` z8 c; U( I( j 2、析取器(>>)5 u4 x+ w$ E2 o) l8 @6 n3 Z) }8 `
3 \0 P* F! o. y( F# C4 V, I 从流中输入数据。比如说系统有一个默认的标准输入流(cin),一般情况下就是指的键盘,所以,cin>>x;就表示从标准输入流中读取一个指定类型(即变量x的类型)的数据。* P+ G- o# D+ S; {- o! D
0 N3 D6 S; b0 Y3 y$ v 在C++中,对文件的操作是通过stream的子类fstream(file stream)来实现的,所以,要用这种方式操作文件,就必须加入头文件fstream.h。下面就把此类的文件操作过程一一道来。) `: \( |1 o( [& i
& h+ n, l: h1 k) ] }
一、打开文件
3 }9 Z$ V7 p D* ~1 u : v2 `( `6 t0 g6 {" r
在fstream类中,有一个成员函数open(),就是用来打开文件的,其原型是:6 b5 l+ ~( j! O8 t3 m
/ H: w. X8 ]3 \ void open(const char* filename,int mode,int access);参数:
' j# o# r/ s$ p: X G, B# I8 A( j, o 7 S# U5 l# j' G! f8 p2 h4 J
filename: 要打开的文件名
) K/ d: s! n! y Q. \
/ h) J' Y2 {* n% ^' Z mode: 要打开文件的方式
* w7 w" \1 Z6 Q3 P; \ , ]9 r. c9 {: B, O" |
access: 打开文件的属性
# ^* e) k. x1 F' Z7 R 6 c' `- i- v3 o! k4 Z, x6 [- j) z
打开文件的方式在类ios(是所有流式I/O类的基类)中定义,常用的值如下:
: `0 F! X' O7 Q; j/ L+ m2 W
5 K0 e! f- q" ]+ j ios::app: 以追加的方式打开文件9 j( o4 ^/ Y. T4 q0 V1 H
+ I' W) \* |; W ios::ate: 文件打开后定位到文件尾,ios:app就包含有此属性' b" E& B1 k( H$ D0 A3 z& E
3 x$ ^3 _+ g) ? i$ _1 G# Z6 w ios::binary: 以二进制方式打开文件,缺省的方式是文本方式。两种方式的区别见前文
5 B" b5 C5 m8 @ & D6 ?& I% k. K P0 h3 S6 g
ios::in: 文件以输入方式打开(文件数据输入到内存)
; w6 o+ a. I m; {; z$ h ; l9 T7 ?% x; T2 g) e4 Y
ios::out: 文件以输出方式打开(内存数据输出到文件)
% {& b) K$ D! h6 i% e
& L% P5 {5 Y. f" |0 ?3 f4 ` ios::nocreate: 不建立文件,所以文件不存在时打开失败
# W1 S T' l+ t) n9 n 7 N# z w! p3 }2 y4 L- J5 W
ios::noreplace:不覆盖文件,所以打开文件时如果文件存在失败 [; ?% b; O* j; K0 n6 c
0 ~) J8 t& ?5 [
ios::trunc: 如果文件存在,把文件长度设为0
" h5 A% G3 j+ X
" }5 B* c5 w, q! c* Q 可以用“或”把以上属性连接起来,如ios::out|ios::binary
! r! J1 [' X9 W, i' j- Q: _& M ]6 V
% ~- B' L5 X, l7 J 打开文件的属性取值是:2 F Y' K p4 p" w) A6 g8 V
4 J6 E$ R6 a* V- O, P7 s( o 0:普通文件,打开访问
# G3 t/ L4 N J% I; _2 I
: d! V$ P+ W6 b$ N/ c* U 1:只读文件
5 ^+ \ }2 N0 B; R' K9 P0 S
: V7 @( k) r% @: k, B0 ~: U 2:隐含文件
; A6 K3 m0 z6 E( O: J$ A; @
! [+ o5 e% t7 J2 S8 v3 q U 4:系统文件
8 k: l& h2 c( x4 x6 b $ L, o% Y7 x' `$ D0 n
可以用“或”或者“+”把以上属性连接起来,如3或1|2就是以只读和隐含属性打开文件。
( Q; G) t6 K( n, L6 g 0 {. Y2 G- C7 V' i' c: u
例如:以二进制输入方式打开文件c:\config.sys
, ^& E( q- S( _3 b @8 I ' A- [2 n5 A' }- }
fstream file1;
# }1 E9 W9 P% e' t& D5 C / {1 o V! R, B$ o' Y. M. ^! T
file1.open("c:\\config.sys",ios::binary|ios::in,0);+ s6 ^% L6 q- u. r
: g0 R" v5 X! N. |# s4 E 如果open函数只有文件名一个参数,则是以读/写普通文件打开,即:7 x+ i+ \4 H) q9 n
( B" [# ^) N# [& @9 G6 Q file1.open("c:\\config.sys"); <=> file1.open("c:\\config.sys",ios::in|ios::out,0);, p& N" [1 X1 [: f* {
/ ^& K3 Z: I. H 另外,fstream还有和open()一样的构造函数,对于上例,在定义的时侯就可以打开文件了:0 p3 Q3 L& D6 ?+ p3 E" e
- w& x8 k9 H, N" P3 s' v% T t5 t
fstream file1("c:\\config.sys"); 特别提出的是,fstream有两个子类:ifstream(input file stream)和ofstream(outpu file stream),ifstream默认以输入方式打开文件,而ofstream默认以输出方式打开文件。
5 C) x7 g! L2 S 9 I: P# k( Q C# t0 X
ifstream file2("c:\\pdos.def");//以输入方式打开文件
6 w9 `3 ^& C9 X8 [8 \- m; P
/ E$ l$ ?7 {6 r ofstream file3("c:\\x.123");//以输出方式打开文件 所以,在实际应用中,根据需要的不同,选择不同的类来定义:如果想以输入方式打开,就用ifstream来定义;如果想以输出方式打开,就用ofstream来定义;如果想以输入/输出方式来打开,就用fstream来定义。
+ d6 j" e5 x, [7 p8 G$ X
6 ~& D+ X$ g0 | 二、关闭文件
3 @, Q/ I! A6 b5 B8 Y " d3 ?: G4 L& ~& i, ]. B
打开的文件使用完成后一定要关闭,fstream提供了成员函数close()来完成此操作,如:file1.close();就把file1相连的文件关闭。
+ ~* O8 \" N$ B. C ~
4 b% p6 b3 ^' K5 y* Q/ } 三、读写文件' ?, k$ Q7 E( H& x$ E$ C
* A+ K' a ~8 W+ D' ~) t
读写文件分为文本文件和二进制文件的读取,对于文本文件的读取比较简单,用插入器和析取器就可以了;而对于二进制的读取就要复杂些,下要就详细的介绍这两种方式
9 h4 i2 \6 W' p+ S6 B5 I2 S) N0 S# `+ y
( h+ k+ \9 a- g% v6 `( a 1、文本文件的读写4 f a* f5 C7 g H
; d0 Y. Y1 a# s/ g% ^
文本文件的读写很简单:用插入器(<<)向文件输出;用析取器(>>)从文件输入。假设file1是以输入方式打开,file2以输出打开。示例如下:
?" d# [3 M$ s
& f; G4 I0 P1 H5 i file2<<"I Love You";//向文件写入字符串"I Love You"% ]3 i, R6 p0 z% x8 }7 R& h' }
2 D6 p( ?+ ?3 ]2 F0 u1 O) [2 B int i;
* x7 O8 |) ]5 |
9 x4 B# f) S! j6 \$ N1 N5 d+ h5 g file1>>i;//从文件输入一个整数值。
6 i7 N6 L" i, A! k1 U) I
Z [9 M+ `% E' Q# t 这种方式还有一种简单的格式化能力,比如可以指定输出为16进制等等,具体的格式有以下一些3 @+ ]% v% p% K/ @# S! i
- X- f G/ W' w3 N2 W+ a# H 操纵符 功能 输入/输出% Z9 g/ Z; g" e2 K E* R
3 X$ L9 k% Z+ h. g2 a dec 格式化为十进制数值数据 输入和输出
3 i! o1 N+ k9 }! y; @8 m/ o
# `6 g. L( V( U$ j' W: X& @ endl 输出一个换行符并刷新此流 输出
9 V: p6 c7 q$ T* {/ T R0 F2 x- } - {( N7 M/ C2 R# n# b
ends 输出一个空字符 输出/ Y2 I- G1 W1 S$ \# l
8 A( W( L% O% y' R& O6 P K hex 格式化为十六进制数值数据 输入和输出" `2 f0 L! l" J1 g+ M" Z, c s0 h
3 f1 H( G' d3 _4 v$ r
oct 格式化为八进制数值数据 输入和输出# A! G! o. F* h& h4 S$ c# R
4 X. W! L" _$ a9 M setpxecision(int p) 设置浮点数的精度位数 输出
3 |. ?6 L$ e9 W z2 I6 D
) ^8 Q3 a. E3 m- c 比如要把123当作十六进制输出:file1<9 a6 C& d a6 h5 H! D
2、二进制文件的读写
2 `9 [2 W8 v4 d+ k$ Z& _+ v
, X& ^1 U( ~& \; Q1 w ①put()
1 @! `& O3 B/ \' l ! v0 R- ]- s( q# l: d" _
put()函数向流写入一个字符,其原型是ofstream &put(char ch),使用也比较简单,如file1.put('c');就是向流写一个字符'c'。1 L/ E# o9 `% z$ e
$ X3 p0 I8 d! L/ _ ②get(); I) I8 D. k$ x' s
1 P* d" C" x/ W7 K2 b8 m get()函数比较灵活,有3种常用的重载形式:/ T" k2 t( E+ ^9 ^# P3 l# B. V
; e( q7 s2 P- B6 q 一种就是和put()对应的形式:ifstream &get(char &ch);功能是从流中读取一个字符,结果保存在引用ch中,如果到文件尾,返回空字符。如file2.get(x);表示从文件中读取一个字符,并把读取的字符保存在x中。0 h+ J4 E3 N9 ~+ q# M
( M# i3 t" b5 J- Z 另一种重载形式的原型是: int get();这种形式是从流中返回一个字符,如果到达文件尾,返回EOF,如x=file2.get();和上例功能是一样的。3 ~& _: v* q/ h7 H# R7 Z, R
3 L% G" a( | l4 O' k: c 还有一种形式的原型是:ifstream &get(char *buf,int num,char delim='\n');这种形式把字符读入由 buf 指向的数组,直到读入了 num 个字符或遇到了由 delim 指定的字符,如果没使用 delim 这个参数,将使用缺省值换行符'\n'。例如:- G9 S$ [5 X4 O2 n7 l1 I
) A8 J" i6 n4 I4 W file2.get(str1,127,'A'); //从文件中读取字符到字符串str1,当遇到字符'A'或读取了127个字符时终止。: o; f9 g# |% I8 G$ B
& K! w( S4 J& K
③读写数据块, M2 Y) c' h% s( w- @
. E4 z ~* r( \& ]; i$ y1 T' ]+ y 要读写二进制数据块,使用成员函数read()和write()成员函数,它们原型如下:. R. z0 o- a. ?
. ?* V' ^: Y7 @% p& b0 E read(unsigned char *buf,int num);
3 m. h3 k$ u% [; _1 h ) s9 [! [% B; H. ~+ T* j4 K
write(const unsigned char *buf,int num);
" Z3 |3 W0 ^1 r( U7 l- F; [
% R- i; D4 }8 W. ] read()从文件中读取 num 个字符到 buf 指向的缓存中,如果在还未读入 num 个字符时就到了文件尾,可以用成员函数 int gcount();来取得实际读取的字符数;而 write() 从buf 指向的缓存写 num 个字符到文件中,值得注意的是缓存的类型是 unsigned char *,有时可能需要类型转换。
C5 [$ Z; y/ P/ C( @5 ~& Y
; p; ^; G7 [1 s% ]! f 例:+ ]! ?* r' X5 U2 E( x
" V3 X- I8 u" @# Q; f unsigned char str1[]="I Love You";
8 g1 J! J. [7 v, ] 1 k4 I% Z! @6 }# d
int n[5];* N. \! W8 }8 S7 M% M+ k) |* p& E
7 y$ M$ r7 R/ V
ifstream in("xxx.xxx");0 O0 |$ l: }/ G6 `/ R: I7 K
2 H, f* x+ W% p' z# [ ofstream out("yyy.yyy");
) V8 i: X# x# @7 p4 b1 M
- |/ |6 K0 x+ C- | out.write(str1,strlen(str1));//把字符串str1全部写到yyy.yyy中
# y0 }/ f: P5 {2 z# [ 9 B' u; Q) V7 S( i
in.read((unsigned char*)n,sizeof(n));//从xxx.xxx中读取指定个整数,注意类型转换
) Q1 `. \, ]* y7 T/ q8 F5 k ) [, S. M" U/ G, Y# J
in.close();out.close(); 四、检测EOF9 I) L( a4 c O$ y3 K% S1 n3 y
+ ^2 [; s: v: ], ]+ Y/ d
成员函数eof()用来检测是否到达文件尾,如果到达文件尾返回非0值,否则返回0。原型是int eof();; R' o# D: K0 j6 L3 |! G! c
+ @6 @& H1 Y; P5 g& E" D
例: if(in.eof()) ShowMessage("已经到达文件尾!");/ i" }$ E4 G6 C6 t! R2 r
1 S! o# [, `' ?* H' Y( w: o 五、文件定位' N8 E; _% A1 Z7 K1 ]
6 }5 T! A8 R5 D2 u. A8 z. I9 H: f) |$ o 和C的文件操作方式不同的是,C++ I/O系统管理两个与一个文件相联系的指针。一个是读指针,它说明输入操作在文件中的位置;另一个是写指针,它下次写操作的位置。每次执行输入或输出时,相应的指针自动变化。所以,C++的文件定位分为读位置和写位置的定位,对应的成员函数是seekg()和seekp()。seekg()是设置读位置, seekp是设置写位置。它们最通用的形式如下:; `' h1 c2 G- I" G
! R$ i8 v, M8 a6 |7 |4 ]
istream &seekg(streamoff offset,seek_dir origin);
3 s2 C+ a ^& G, ?/ I/ o; J1 @/ { 5 n5 g# O! o& v" l- _: l3 S
ostream &seekp(streamoff offset,seek_dir origin);8 }6 V* n/ S7 Q3 h: F v
+ i( v% Q* m. A' h+ g streamoff定义于 iostream.h 中,定义有偏移量 offset 所能取得的最大值,seek_dir 表示移动的基准位置,是一个有以下值的枚举:
' h( K6 O' y( w. M# {% u3 l
0 t; C( Z: G# W4 E' Y ios::beg: 文件开头5 q# i) E* ]1 C! Q0 K2 t5 F+ u
# ~: z' W3 b/ i9 Q, U ios::cur: 文件当前位置
/ k, {6 n: K0 u# g& _2 X$ }4 x + l! K* q9 e2 e: R1 v) O/ x
ios::end: 文件结尾. F4 x( U- \5 M" N, z# l
9 J/ K( O' O, x) x7 ]( K
这两个函数一般用于二进制文件,因为文本文件会因为系统对字符的解释而可能与预想的值不同。例:
# D9 R) |; z) s0 _# Y$ E3 R* `
6 S. G2 ?) R4 b% w# L file1.seekg(1234,ios::cur); //把文件的读指针从当前位置向后移1234个字节
# L5 O- P3 [$ O" I
) {; V1 k. O' D4 t! B" @% N4 ?7 f file2.seekp(1234,ios::beg); //把文件的写指针从文件开头向后移1234个字节/ b7 i. S8 B6 C' \% {
, q5 s' P- S4 ~9 A: e* p2 j |
|