PLM之家PLMHome-国产软件践行者

[转载电子书] ifstream 和 ofstream的用法详细介绍

[复制链接]

2014-1-23 14:57:18 4395 0

admin 发表于 2014-1-23 14:57:18 |阅读模式

admin 楼主

2014-1-23 14:57:18

请使用QQ关联注册PLM之家,学习更多关于内容,更多精彩原创视频供你学习!

您需要 登录 才可以下载或查看,没有账号?注册

x
ofstream是从内存到硬盘,ifstream是从硬盘到内存,其实所谓的流缓冲就是内存空间;* R0 ^# L% O  @3 A' _
) }! y8 S; p* O
  在C++中,有一个stream这个类,所有的I/O都以这个“流”类为基础的,包括我们要认识的文件I/O,stream这个类有两个重要的运算符:
0 t: t) W$ y6 n2 r& R1 c % F- @& U. t3 Y$ S# }  C& }. s$ y
  1、插入器(<<)# s( h, u, t* _6 A7 }1 h- d( ^  h
+ }. U1 v3 j  V- h7 Y
  向流输出数据。比如说系统有一个默认的标准输出流(cout),一般情况下就是指的显示器,所以,cout<<"Write Stdout"<<'\n';就表示把字符串"Write Stdout"和换行字符('\n')输出到标准输出流。+ Y6 M8 @0 y" N8 ?
. l4 c) H5 v( m: e0 E& Y
  2、析取器(>>)
" c) i% w/ z% i$ k8 g: l8 s 1 J9 f- k0 J! R# ?/ A
  从流中输入数据。比如说系统有一个默认的标准输入流(cin),一般情况下就是指的键盘,所以,cin>>x;就表示从标准输入流中读取一个指定类型(即变量x的类型)的数据。
/ L! z/ J5 l0 N( H. R3 | 8 o6 g9 v! g% M
  在C++中,对文件的操作是通过stream的子类fstream(file stream)来实现的,所以,要用这种方式操作文件,就必须加入头文件fstream.h。下面就把此类的文件操作过程一一道来。2 A7 N6 k, K9 z/ _
/ ]) X2 S) J9 @9 i4 x: O
  一、打开文件
; T1 O( y7 v3 n( j / l1 U- o+ R) s. e, K# j( |% y
  在fstream类中,有一个成员函数open(),就是用来打开文件的,其原型是:
0 Q% M2 A9 c6 Q' F. d
1 y  o$ R! v" u1 F  void open(const char* filename,int mode,int access);参数:1 ]! f4 U5 p0 y* s# T! N) p2 t/ S
1 T/ u9 j- P& O
  filename:  要打开的文件名& m  [* M% R" w$ `- V

4 e* g3 {$ I0 k# Q# Q" t$ L: D: T  mode:    要打开文件的方式, Y- a$ O) Q, m
7 e1 N3 [! H3 q* x# s# B# N* S
  access:   打开文件的属性
$ v- }& `9 e* I! i + }4 d0 m/ i# N. K, t0 `/ O( W* c
  打开文件的方式在类ios(是所有流式I/O类的基类)中定义,常用的值如下:
3 d! j" j6 w* _9 o& A: a9 W  `
) ]$ s7 a8 ?' w9 F3 W3 s  ios::app:   以追加的方式打开文件# P: ?+ E+ F9 T0 |  s
# E7 K, Z' ~4 r9 g) B8 M) O9 R
  ios::ate:   文件打开后定位到文件尾,ios:app就包含有此属性
$ z' G5 p$ J% p* F8 G& F
: f- W& p% r. T$ n+ m0 x  ios::binary: 以二进制方式打开文件,缺省的方式是文本方式。两种方式的区别见前文+ R0 h/ r2 r* |+ o
( p- A; S8 l, j& S
  ios::in:    文件以输入方式打开(文件数据输入到内存)
+ h% t$ d% z! N* L' D9 }9 j
4 h; d% ?/ P* ~# G/ J3 Q6 I  ios::out:   文件以输出方式打开(内存数据输出到文件)
5 E9 ?- f2 d3 M& ^
. Y# e* {" W& I& E, L  ios::nocreate: 不建立文件,所以文件不存在时打开失败
0 N" ^% y4 D1 j% S 4 O4 W' i% ^1 M! g- c
  ios::noreplace:不覆盖文件,所以打开文件时如果文件存在失败$ J& h1 z4 F, c1 W# M8 k
: w4 e+ G/ X5 O
  ios::trunc:  如果文件存在,把文件长度设为0
" p4 h$ i0 K/ [$ q) m
% v- s; D$ A, ]! v) G  可以用“或”把以上属性连接起来,如ios::out|ios::binary
8 c$ c' J9 v, W4 z0 J1 m& A2 S
, F9 }9 m: Q) Y+ k  打开文件的属性取值是:6 `% {; |- R- Q/ n# d

( O3 B( x; w9 y8 l; s0 j! ^  0:普通文件,打开访问
$ l6 M+ P) |# a$ x6 b1 |/ G
2 }& B$ s7 K6 M7 B3 k5 P1 q  1:只读文件
: k9 F( {8 k7 a& @
; f# `- Y8 q8 A% o  2:隐含文件# ^$ j: c  s1 n" b: {

# R# ^# D: l) k1 l$ a% U. q  4:系统文件
7 [8 B$ o" p' b. e+ F+ @
" ~: P; c  Y  K4 S7 G  可以用“或”或者“+”把以上属性连接起来,如3或1|2就是以只读和隐含属性打开文件。  X/ m9 _0 j$ Z( z
0 H9 ?# B/ O: r; K( ~% ^( c
  例如:以二进制输入方式打开文件c:\config.sys: A$ _. ^7 t& Z; u; D9 R" t, r

/ @1 s6 e. |& _8 P$ r  M  fstream file1;2 ?. Z1 z+ B$ z# H2 a2 v

7 Y& @* h$ @1 C% P  E3 L  file1.open("c:\\config.sys",ios::binary|ios::in,0);
9 ^7 M: [# t; V' f  g1 b% G' E2 T $ \6 Y) V* \+ y
  如果open函数只有文件名一个参数,则是以读/写普通文件打开,即:
& m8 Z/ N4 l. f 9 ]% B* P* |# ]3 }2 F4 \2 I  B8 S
  file1.open("c:\\config.sys"); <=> file1.open("c:\\config.sys",ios::in|ios::out,0);
' {, W1 d6 R  H+ Z7 W  J
7 a' K1 n  V; b+ i1 S! K  另外,fstream还有和open()一样的构造函数,对于上例,在定义的时侯就可以打开文件了:
( @& K6 A- [  s  C. S * U+ j& ~( ~2 j/ B" b. |8 _7 r$ V) ?
  fstream file1("c:\\config.sys");  特别提出的是,fstream有两个子类:ifstream(input file stream)和ofstream(outpu file stream),ifstream默认以输入方式打开文件,而ofstream默认以输出方式打开文件。1 o/ j; P. q7 l& k* \( }

9 F- c3 o3 w" G) V- T  ifstream file2("c:\\pdos.def");//以输入方式打开文件
6 n4 b8 c. E7 O- m* S . \% D7 i: s8 R& P) \% ]
  ofstream file3("c:\\x.123");//以输出方式打开文件  所以,在实际应用中,根据需要的不同,选择不同的类来定义:如果想以输入方式打开,就用ifstream来定义;如果想以输出方式打开,就用ofstream来定义;如果想以输入/输出方式来打开,就用fstream来定义。
( s1 _3 P& A0 f4 o, q
- g4 g: h+ n7 n' F+ X  二、关闭文件( i- V( ~/ j5 C7 v

9 g: J: b. z- \/ f5 x* ^  打开的文件使用完成后一定要关闭,fstream提供了成员函数close()来完成此操作,如:file1.close();就把file1相连的文件关闭。
" k  Y6 |/ \$ L) z. ?  _; G' I " i' _# v5 e- i" H6 q
  三、读写文件" g. s$ p1 |2 w/ X" s! `8 \1 L
- K: O$ C! `: a: M+ ?' M
  读写文件分为文本文件和二进制文件的读取,对于文本文件的读取比较简单,用插入器和析取器就可以了;而对于二进制的读取就要复杂些,下要就详细的介绍这两种方式
2 ^" m/ M4 J$ Z
1 Y1 R' @% S; }* o3 b4 Y# }, C  1、文本文件的读写
+ i% s8 h! b4 c! A1 b- K! P' Y& H8 I
$ z, }9 H* N5 `- i) A3 N. b  文本文件的读写很简单:用插入器(<<)向文件输出;用析取器(>>)从文件输入。假设file1是以输入方式打开,file2以输出打开。示例如下:0 p+ M& m( L/ _" |
: z) x8 H$ n/ l: Y$ c+ D
  file2<<"I Love You";//向文件写入字符串"I Love You"6 h1 I" z+ z$ ]- O5 V- H! V8 U" o
; ^4 p( m9 v, B5 S) k6 L1 g
  int i;
# O, u. H7 P/ O0 u, N4 t 6 l& B7 m( B5 a4 I' N
  file1>>i;//从文件输入一个整数值。
; M1 H; S: |. ~$ f) @1 {! H$ W 8 S  G/ O. o2 p: ]; c" d0 B
  这种方式还有一种简单的格式化能力,比如可以指定输出为16进制等等,具体的格式有以下一些
& {. A+ U' u3 g1 r8 P, y 3 S* k; z& J9 e; k5 K; s/ D( B
  操纵符 功能 输入/输出
8 p) ^. q0 c9 C! M1 c7 Y) M( z 8 M1 ~: e9 U) r) @* B) N+ W, {8 s
  dec 格式化为十进制数值数据 输入和输出
' q# u+ S" ~3 l9 }: Q- i- Q, P % K) N5 H  ?/ F! `6 |4 R
  endl 输出一个换行符并刷新此流 输出# N! j4 P, h+ j; @" P! \# R
7 L0 n6 {, [& m4 z8 K! r( p( P* |
  ends 输出一个空字符 输出1 G: g( |  d2 {! ?: _- \. i

* v, s4 h/ b- \/ x5 N. O  hex 格式化为十六进制数值数据 输入和输出  v0 o% [. P  p( t6 l2 Z, I( @0 E" b

6 H* ~# N* I. D& j  oct 格式化为八进制数值数据 输入和输出+ \6 o4 K* |" R0 q2 y, k% g
- C8 Y7 ~3 X0 Y3 X% q6 j
  setpxecision(int p) 设置浮点数的精度位数 输出
1 A9 O" K4 B* l  R$ K* W) V% i 0 ?# m0 w% B! T# [  ?' \' g- D
  比如要把123当作十六进制输出:file1<

* g3 d* Z# s9 S8 L* F) y/ F0 k+ a$ c  2、二进制文件的读写
. S4 A$ R7 n9 M
, t+ \- a& X6 b$ [; a. K+ p  ①put()" a5 a; h8 ^' d# ?
' i; w( B& ^+ f
  put()函数向流写入一个字符,其原型是ofstream &put(char ch),使用也比较简单,如file1.put('c');就是向流写一个字符'c'。, N2 _6 C5 U$ s5 {$ T, a1 B" H  W
/ H) I/ t$ n5 s" z1 C& _1 U
  ②get()
8 s2 c1 O* F7 a, K* z) n9 u" o% @ ' W3 @" R/ A4 a* X+ W3 l5 I
  get()函数比较灵活,有3种常用的重载形式:- S/ G) ?; b* x6 I

( U4 U2 r7 I6 Q$ Y  o+ w( B  一种就是和put()对应的形式:ifstream &get(char &ch);功能是从流中读取一个字符,结果保存在引用ch中,如果到文件尾,返回空字符。如file2.get(x);表示从文件中读取一个字符,并把读取的字符保存在x中。  E2 V. }' a5 {0 X! M
' n% A& R1 @; D0 r7 ?8 S* {
  另一种重载形式的原型是: int get();这种形式是从流中返回一个字符,如果到达文件尾,返回EOF,如x=file2.get();和上例功能是一样的。- I2 ^, l: G. O% E1 G! H
; V0 ]" P6 y  J* q( ~4 f) d9 y
  还有一种形式的原型是:ifstream &get(char *buf,int num,char delim='\n');这种形式把字符读入由 buf 指向的数组,直到读入了 num 个字符或遇到了由 delim 指定的字符,如果没使用 delim 这个参数,将使用缺省值换行符'\n'。例如:5 ^% O2 [( y) Q7 J8 {
- i- L) v4 O$ A  X$ b; K4 x
  file2.get(str1,127,'A'); //从文件中读取字符到字符串str1,当遇到字符'A'或读取了127个字符时终止。
* @6 x" J9 Z8 O: \
5 Q7 M% j  @4 \  ③读写数据块+ p0 ~$ E5 J  P' B

/ {: w' O' z1 M+ R  要读写二进制数据块,使用成员函数read()和write()成员函数,它们原型如下:8 w; I& p1 y2 f8 @: }
* v% M6 `3 j9 n) P
  read(unsigned char *buf,int num);6 q' v* l9 T) _) j' |; e
, s* e" E" G, K$ |
  write(const unsigned char *buf,int num);
8 ?0 M% w! u: d- O
) V, k, W0 ~% @4 V) d" z8 q  read()从文件中读取 num 个字符到 buf 指向的缓存中,如果在还未读入 num 个字符时就到了文件尾,可以用成员函数 int gcount();来取得实际读取的字符数;而 write() 从buf 指向的缓存写 num 个字符到文件中,值得注意的是缓存的类型是 unsigned char *,有时可能需要类型转换。/ e+ P: O: m0 y9 m4 S5 ?( w

8 `+ K4 b$ g$ t3 S  P  例:0 M5 [% ?; e5 R; I/ C: I4 ?; c

# D+ ]4 ]/ M. J/ s  unsigned char str1[]="I Love You";
7 j! H: C2 U+ U, x! O" d* h" ?2 l
/ F3 l6 ^, w7 E' E9 [  int n[5];
* ?- U+ L- Y2 ~) C( h. \- \ 8 i" h5 l4 [4 s3 _0 f9 h8 L
  ifstream in("xxx.xxx");' Z9 H! l9 u0 ?2 r5 h* s

  m7 @% T1 C; v$ \7 m. t* |* q' T( x  ofstream out("yyy.yyy");* N" S& r& P: M, W
& F' f$ W, K) ?- F4 D5 f, K
  out.write(str1,strlen(str1));//把字符串str1全部写到yyy.yyy中
( }' ]9 m4 Z: z' C 9 I# F& N; a$ U. d' m9 X/ ^7 H. K
  in.read((unsigned char*)n,sizeof(n));//从xxx.xxx中读取指定个整数,注意类型转换
$ W7 Z5 x6 _$ I( S9 p8 ~* h & u" [% O* b- `1 Y' W1 j& f
  in.close();out.close(); 四、检测EOF
+ H7 w6 C' Z% Q6 F0 O / i; n# N: |2 E2 E, h" V
  成员函数eof()用来检测是否到达文件尾,如果到达文件尾返回非0值,否则返回0。原型是int eof();7 c) i$ J6 N+ X

9 O& [3 c: H* g  例:  if(in.eof()) ShowMessage("已经到达文件尾!");
0 X0 ]  ~+ k1 |, t" `) i  f
! ~) K' ~6 ?1 q' D1 J' M  五、文件定位0 ^0 ~& y+ ]" p4 J- G

) \! x* B! q" r  {9 j# Z% r" V0 [/ D  和C的文件操作方式不同的是,C++ I/O系统管理两个与一个文件相联系的指针。一个是读指针,它说明输入操作在文件中的位置;另一个是写指针,它下次写操作的位置。每次执行输入或输出时,相应的指针自动变化。所以,C++的文件定位分为读位置和写位置的定位,对应的成员函数是seekg()和seekp()。seekg()是设置读位置, seekp是设置写位置。它们最通用的形式如下:: [/ Y( ?' P8 C* t; V7 q8 M
8 O% F9 d! t4 _+ X# ?
  istream &seekg(streamoff offset,seek_dir origin);! @/ `' d$ a0 a' y

: b4 o# f* W" f  ostream &seekp(streamoff offset,seek_dir origin);
6 ~- B) P; M. ^  C ) n% H; s7 c# \+ x. X
  streamoff定义于 iostream.h 中,定义有偏移量 offset 所能取得的最大值,seek_dir 表示移动的基准位置,是一个有以下值的枚举:2 Q- ?! N9 E2 k! n# [0 W+ }
# A4 G) q8 w* b, f3 [
  ios::beg:  文件开头
$ p% Z) l  u# e0 i
: L# z& O6 B5 _/ C3 W* P  ios::cur:  文件当前位置
( S9 G) `( ^0 [ # L. v- r5 a; l! p' L6 ~
  ios::end:  文件结尾
/ Y4 t5 Y: f) b5 r" b; I: |- K
* p, p! @2 s5 e) i# \% }  这两个函数一般用于二进制文件,因为文本文件会因为系统对字符的解释而可能与预想的值不同。例:
4 O! S% S) h: ?. E
* T' \6 c1 x4 R. _. s  v3 g5 ]  file1.seekg(1234,ios::cur); //把文件的读指针从当前位置向后移1234个字节$ Y) n0 c5 l( S4 v
. T0 |& E- l2 z9 U$ o
  file2.seekp(1234,ios::beg); //把文件的写指针从文件开头向后移1234个字节
, i# a' |7 r: H0 k& i( R/ {

8 I- x+ X* `! p3 H
上海点团信息科技有限公司,承接UG NX,CATIA,CREO,Solidworks 等CAx软件,Teamcenter,3D Experience等PLM软件,工业4.0数字化软件的实施\二次开发\培训相关业务,详情QQ 939801026 Tel 18301858168 网址 doTeam.tech
回复

使用道具 举报

发表回复

您需要登录后才可以回帖 登录 | 注册

返回列表 本版积分规则

  • 发布新帖

  • 在线客服

  • 微信

  • 客户端

  • 返回顶部

  • x
    温馨提示

    本网站(plmhome.com)为PLM之家工业软件学习官网站

    展示的视频材料全部免费,需要高清和特殊技术支持请联系 QQ: 939801026

    PLM之家NX CAM二次开发专题模块培训报名开始啦

    我知道了