C++中IO流

一、使用流

 1、到底什么是流

   cout和cin都是在C++的std命名空间中预定义的流实例。

   每个输入流都有一个相关联的源,每个输出流都有一个相关联的目的。

 2、流的源和目的

   控制台流、文件流、字符串流,还有比如打印机输出或网络I/O是由操作系统提供的,并没有内置到C++语言中。

 3、流输出

   ①、cout流是把数据写到控制台或者标准输出的内置流。

   ②、C++流可以正确地解析C风格的转义代码,比如包含\n的字符串,但是这样的情况下,内置的endl机制会更好。使用endl用于表示行结束字符并刷新输出缓冲区。

   put()和write()是原始的输出方法,它们是cout提供的公共方法。

   向输出流写数据时,流不必立刻把数据写到目的中,大部分的输出流都会进入缓冲区,或累积数据,而不是数据一到来就写出。当下列条件之一发生时,流会刷新输出,或者写出累积的数据:

·      到达一个标记,比如endl标记。

·      流超出作用域,因此被撤销。

·      对应的输入流请求输入(也就是,cin输入时,cout会刷新输出)

·      流缓冲区已满

·      明确告诉流要刷新输出其缓冲区(flush()方法)

   注:不是所有的流都会缓冲,cerr流就没有对其输出进行缓冲。

  

①、处理输出错误

·      good()函数:在流上直接调用good()来确定流当前是否是好的(或的流的有效性基本信息)

·      bad()函数:返回true表示发生了一个致命的错误。

·      fail()函数:如果最近的操作失败,它会返回true,意味着下一个操作也会失败。

·      clear()函数:重置流的错误状态。

②、输出控制符

 控制符是一些对象,他们可以改变流的行为。

 <iomanip>:setw和setfill、<iostream>头文件包含。

 

 4、流输入

   输入流提供了一种读取结构化或非结构化数据的简单方法。

   和输出流一样,输入流也提供了一些方法来完成更底层的访问:

·      get():允许从流输入原始数据。

·      unget():unget()调用会引起流后退一个位置,其本质是把最后一个字符读回到流中。

·      puthack():允许在输入流中后退一个字符,putback()方法取流中要后退的字符作为参数。

·      peek():peek()方法允许预览下一个值。

·      getline():方法getline()使用一行不超过指定长度的数据来填充一个字符缓冲区,所得到的字符串不包括换行字符。

    处理输入错误

    输入控制符

    例:boolalpha、noboolapha、hex、oct、dec、skipws、noskipws、ws。

 

5、 输入和输出对象

 在C++中,对象可以预定如何进行输入输出,通过重载操作符<<,使之了解如何输出一个新类型或类来做到。

 

二、字符流

 流具有内置的词法分析功能,所以字符串流对于解析文本也很有用处。

 类ostringstream和istringstream分别用于向字符串写数据和从字符串读数据。它们都定义在头文件<sstream>中。 

 

三、文件流

 文件非常适合于流抽象,因为除了数据之外,读写文件总是会涉及到位置。

 C++中,类ofstream和ifstream提供了文件的输入输出功能,定义在头文件<fstream>中。

 参考代码: 

/**//*

 *NOTE: file_stream.cpp - file stream test...

 * 

 */

 

#include <fstream>

#include <string>

#include <iostream>

using namespace std;

int main(int argc, char *argv[])

...{

    /**//* create file stream. */

    ofstream fout("file_stream.txt");

    

    if (!fout.good()) ...{

        cerr << "error opening file_stream.txt for writing. ";

        exit(1);

    }

    

    /**//* Output the string "name = lulixue" */

    fout << " [name]=lulixue";

    

    /**//* Verify that the marker is at the end. */

    streampos curPos = fout.tellp();

    

    if (curPos) ...{

        cout << "Test passed: Currently at position 6." << endl;

    } else ...{

        cout << "Test failed: Not at postion 6." << endl;

    }

    

    /**//* Move to position 2 in the stream */

    fout.seekp(2, ios_base::beg);

    

    fout << 'N';

    fout.flush();

    

    ifstream fin("file_stream.txt");

    

    if (!fin.good()) ...{

        cout << "Error opening file_stream.txt for reading. ";

        exit(1);

    }

    

    /**//* Read the file string */

    string file_str;

    fin >> file_str;

    string strMacth = "[Name]=lulixue";

    cout << "strMacth: " << strMacth << endl;

    cout << "file_str: " << file_str << endl;

    if (file_str == strMacth) ...{

        cout << "Test passed: Value is [Name]=lulixue" << endl;

    } else ...{

        cout << "Test failed: Vlaue is not [Name]=lulixue";

    }

    

    return 0;

}

 

   Ø    链接流:可以在任何输入流和输出流之间建立连接,也即,从输入流请求数据时,与其链接的输出流会自动刷新输出。

   Ø    链接流用方法tie()来实现。要把输出流绑定到一个输入流中,可以在输入流上调用   tie(),并把输出流的地址传递给它。要断开这个连接,传递NULL即可。

   Ø    也可以把一个输出流链接到另一个输出流上。

   例:outFile.tie(&antherOutFile);,这样,每次向第一个文件写数据时候,就会向另一个文件写入已经发送的缓冲数据,可以用这种机制保持两个相关文件之间的同步。

 

四、双向I/O

   有一种流可以同时进行输入输出,双向流可以同时作为输入流和输出流操作。

   双向流是iostream的子类,所以也是istream和ostream的子类,因此可以作为一个有用的多重继承的例子。双向流同时支持>>和<<操作符,还支持输入流和输出流方法。

   fstream类提供了一个双向的文件流,如果应用需要替换一个文件中的数据,fstream就非常理想,因为找到正确的位置之前可以一直完成读操作,找到之后立即切换为写操作。

   只有数据长度固定时,这样的方法才能正常工作。

   同过string stream类,也可以以双向方式访问字符串流。

   注:双向流对于读位置和写位置分别由单独的指针,在读操作与写操作之间切换时,需要查找到正确的位置。 

   实例代码: 

/**//*

 * bidirectional_stream.cpp - bidirectional file stream test...

 */

#include <iostream>

#include <fstream>

#include <string>

using namespace std;

void change_number_for_id(const string& in_file_name,  int id, const string& in_new_number)

...{

    cout << "3" << endl;

    fstream io_data(in_file_name.c_str());

    if (!io_data) ...{

        cerr << "Error while opening file " << in_file_name << endl;

        exit(1);

    }

    while (io_data.good()) ...{

        int id_feild;

        string number;

        //Read the next ID

        io_data >> id_feild;

        //Check to see if the current record is the one being changed.

        if (id_feild == id) ...{

            //Seek to the current position.

            io_data.seekp(io_data.tellg());

            //Output a space, then the new number

            io_data << " " << in_new_number;

            break;

        }

        //Read the current number

        io_data >> number;

    }

    cout << "4" << endl;

}

int main(int argc, char *argv[])

...{

    string str_file_name = "bidirectional_input.txt";

    string str_number    = "111-222-3333";

    cout << "1" << endl;

    change_number_for_id(str_file_name, 333, str_number);

    cout << "2" << endl;

    return 0;

posted @ 2012-07-19 23:31  dancingrain  阅读(448)  评论(0编辑  收藏  举报