程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> C語言 >> C++ >> C++入門知識 >> Redirect the stream buffer of cout to a file with ios::rdbuf

Redirect the stream buffer of cout to a file with ios::rdbuf

編輯:C++入門知識

Introduction:
• This article mainly about how to use ios::rdbuf() function redirect the standard output stream to a file ,that is,redirect the cout output stream.
• Two questions will be answered in this article: 1.how to redirect a stream. 2. what is the real face of cout,cin and  cerr.
     很多時候,我們為了實現一些特殊的目的,都會涉及到給控制台輸出重定向的問題,比如在編寫GUI程序時,為了調試方便,我們需要把一些中間結果輸出,但如果用MessageBox,
真是差強人意,輸個字符串還湊合,但如果要輸出個數值或者地址,我們就不得不創建一個CString對象將它們格式化成一個字符串再用MessageBox輸出,這可真辛苦!當然每個人的
調試方法都不一樣,各有絕招。本文主要講述如何將 cout輸出流重定向到其他地方,比如一個文件,當然如果你想在自己的GUI工程中把一些調試數據通過cout輸出到一個文件,或者
直接用cout寫日志,那麼本文將會對你有幫助,其次如果當你了解了重定向就、技術後,你將會重新認識一些老朋友比如:cout, cin, cerr的真正身份。
    首先我們介紹一位 basic_ios 類中非常重要的一位成員,他就是ios::rdbuf().我們先來看看這個函數的定義(vs2008):
     typedef basic_streambuf<_Elem, _Traits> _Mysb;
    _Mysb *rdbuf() const
        {   // return stream buffer pointer
        return (_Mystrbuf);
        }
     _Mysb * rdbuf(_Mysb *_Strbuf)
        {    // set stream buffer pointer
        _Mysb *_Oldstrbuf = _Mystrbuf;
        _Mystrbuf = _Strbuf;
        clear();
        return (_Oldstrbuf);
        }
不難看出rdbuf()函數正是設置和修改io流指針的,正因為這個原因,我想到是否能通過修改io類關聯的緩沖區指針對其輸入輸出進行重定向,當然事實是樂觀的。作為示例,我對cout動
手術。首先,把cout的輸出重定向到一個文件中,通過以下代碼實現:
     ofstream dwout( "redirection.txt" );
     ofstream::_Mysb * org_cout= cout.rdbuf(dwout.rdbuf());
這樣一來,就成功把cout的輸出緩沖區流重定向到了"redirection.txt"文件,而不會輸出到控制台,當然我們應該把原來流指針保存到 org_cout 裡,用於恢復cout的流緩沖區。接下來
我們可以這樣:
     for (int i = 0; i < 5 ; i++)
      {
          cout<<"redirection ";
      }
向"redirection.txt"中輸入5個"redirection "成功後會輸出到文件而不是控制台。接下來就是打開"redirection.txt"文件,把文件內容輸出到屏幕上,所以我們必須先恢復cout流,使它
輸出恢復到控制台,然後來驗證我們的重定向是否成功,代碼如下:
     dwin.open( "redirection.txt");
     cout<<dwin.rdbuf(); //這條語句可以直接輸出"redirection.txt"的全部內容,這是rdbuf()的另一個強大功能。
下面是完整驗證代碼,第一次打開文件寫 5個"duwen",然後輸出是為了驗證cout<< file.rdbuf()可以將文件所有內容一次性全部輸出,緊接著是重定向cout,然後寫cout 5個
"redirection",緊接著恢復cout流,將文件內容輸出到控制台,由於默認打開方式是默認out,所以第二次打開文件寫時(cout)會把文件內容先清空,也就是把5 個"duwen”清空,所以
第二次顯示後,只顯示5個"redirection" :
    
/************************************************************************
  Description :  Redirect the cout stream.
  Notices      :  Copyright (c) Duwen
  TIME         :  Post time: 5/11/2012, write time:6/3/2011
************************************************************************/

#include "stdafx.h"
#include <iostream>
#include <fstream>
using namespace std;

int _tmain(int argc, _TCHAR* argv[])
{
    //create a redirection file, and write 10 "duwen" in.

    ofstream dwout("redirection.txt");
    for (int i=0 ; i< 5 ; i++)
    {
        dwout<<"duwen  ";
    }
    dwout<<'\n';
    dwout.close();

    //Output the file contents
    ifstream  dwin("redirection.txt");
    cout<<dwin.rdbuf();//Note: this sentence can output whole file
    dwin.close();

   //Open  "redirection.txt", we'll redirect the stream buffer of "cout" latter.
    dwout.open("redirection.txt");
    //Redirect
    ofstream::_Mysb * org_cout=cout.rdbuf(dwout.rdbuf());
    for (int i = 0; i < 5 ; i++)
    {
    //We output 5 "redirection" which cannot  be shown on console, instead, it will be redirected to "redirection.txt".
        cout<<"redirection ";
    }
    dwout.close();
  //open redirection.txt.
    dwin.open("redirection.txt");
  //recover the stream buffer of cout ,that is, console
    cout.rdbuf(org_cout);

  //verify  if our redirection succeeded.
    cout<<dwin.rdbuf();
    cout<<endl;
  //Note: we mustn't close all the files we have opened, because the destructor of each file class did it automatically.
    return 0;
}
好了,有了上面的了解,你是否還能聯想到什麼呢,探索永遠都不會停止.
在我剛開始學習C++時,總是對cout,cin,cerr充滿著好奇心,書上說它們都是一些全局流對象,就拿cout來說,書上說它是ostream的對象,但我試圖這樣 ostream myout 創造出我的 "cout”,時
編譯器是不會放行的,當然,首先能想到的原因就是ostream沒有這樣的無參構造函數,於是我打開 ostream 文檔,裡面有兩個構造函數,如下:
  
 explicit  basic_ostream(basic_streambuf<_Elem, _Traits> *_Strbuf,bool _Isstd = false)
 
 basic_ostream(_Uninitialized, bool _Addit = true)

 
可是當時的我還不能看懂這個聲明的意思,於是我就想著去看看cout的定義,於是我就點擊cout-轉到定義,得到的是__PURE_APPDOMAIN_GLOBAL extern _CRTDATA2 ostream cout;然後我把整個ostream文件搜了個遍,但還是沒找到,說了是全局對象,但又怎麼也找不到,真欲抱琵琶半遮面,千呼萬喚不出現,當時我用了一種變通的方法創造出了我的myout, 怎麼做的呢,你猜猜吧,呵呵,言歸正傳.其實我後來挖掘到這裡的時候,產生了一個想法,仔細看我上面的代碼,我是先創建一個文件,然後把cout的流重定向到文件,再來看看ostream的第一個構造函數,我笑了,你想到了嗎...下面直接給出代碼:
//my cout
ofstream console(stdout);
ostream mycout(console.rdbuf());
//my cin
ifstream  input(stdin);
ostream myin(input.rdbuf());
//my cerr
ofstream error(stderr);
ostream mycerr(error.rdbuf());
好了,本文到此也應該收筆了, 需要提醒讀者的是,io類族中除過ios_base類外其他的都可以用rdbuf進行重定向,而不僅限於cout,本文只是以其示例.還有就是通過重定向可以方便實現其它很多功能,
比如用一個cin語句就可以把鍵盤輸入寫道文件等等,讀者應該能舉一反三.最後,若復制轉載請注明原作者,請支持原作.
 

 

摘自 不在浮沙築高台

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved