2014-02-08 51 views
16

如果我選擇創建Windows非控制檯應用程序,並在代碼中實現printf\cout,那麼printf\cout在哪裏寫入?它是否寫入stdout緩衝區?如果是的話,有沒有辦法從stdout中讀取它並將其打印到一些文本文件中,或者使用文本進行MessageBox(只是爲了驗證我是否正確讀取了它)?「printf」在Windows非控制檯應用程序中寫入的位置?

編輯::只是爲了澄清,我不想在任何地方重定向輸出。我想知道,printf/cout寫在哪裏?如果它寫入某個默認緩衝區,是否有讀取輸出的方法,只是爲了驗證我已經從正確的緩衝區讀取了正確的輸出。 請不要給我解決方案重定向「stdout」

+4

技術上講,它不會在任何地方寫,但你可以通過使用'freopen函數重定向到文本文件(「file.txt的」,「W」,標準輸出) ;' – Predelnik

+1

可能的重複[如何將stdout重定向到Windows應用程序中的某個可見顯示?](http://stackoverflow.com/questions/573724/how-can-i-redirect-stdout-to-some-visible -Display-in-a-windows-application) –

+1

@MartinR ::我不想重定向輸出。我想知道,如果沒有重定向,它在哪裏寫入,並且是否有讀取該位置的方法來驗證。 – Abhineet

回答

13

在Windows下stdout是訪問由GetStdHandle(STD_OUTPUT_HANDLE)返回的句柄的低級函數的包裝。

當通過雙擊啓動非控制檯應用程序(我在Windows 7,64位下試過)時,GetStdHandle(STD_OUTPUT_HANDLE)將返回無效句柄。這意味着printf和其他命令根本不會寫任何內容,但printf內部調用的低級函數將返回錯誤代碼。

但是前面已經說了,甚至一個非控制檯程序可以啓動通過以下方式:

program_name > output_file.txt 

在這種情況下,printf輸出將被寫入到輸出文件。

- 編輯 -

如果要「抓住」 printf()輸出,並將其寫入MessageBox()有兩種方法來實現這一目標:

第一個正在運行的程序的兩倍,而一個實例的輸入是另一個實例的標準輸出。該行爲可以通過以下命令行解釋:

program_name | program_name /msgbox 

第二種可能工作沒有運行兩個程序沒有運行該程序兩次:您可以勾文件句柄#1。這應該使用MSVCRT.DLL時至少可能:

HANDLE hRead,hWrite; 

CreatePipe(&hRead,&hWrite,NULL,0); 
dup2(_open_osfhandle(hWrite,O_WRONLY),1); 

// Now printf() output can be read from handle hRead 

printf("Hello world 2!\n"); 

// In a separate thread do: 
ReadFile(hRead,...); 
+0

+1遠非如此,您的解決方案看起來最適合我。 – Abhineet

2

printfcout總是打印到stdout

您需要從命令行啓動程序並將其輸出傳輸到文本文件以使其可讀。

否則,您需要將輸出流傳輸到tmp中的文件以在您的代碼中生成。

管道可以做下面的方式(如果您的應用程序的名稱將是富):

foo > log.txt 

然後,您可以讀取文件log.txt如果你導航到其目錄可與

dir 
被發現
+0

'printf or cout總是打印到stdout'的語句顯然是錯誤的 –

+0

它們正在打印到標準輸出,您必須重定向作爲調用者 – nesreka

+4

@Dieter真的嗎? 「對象cout控制輸出到與對象stdout相關的流緩衝區」(27.4.2/3) –

4

我正在開發一個可視化引擎,我使用以下兩件事來替代PC和用戶之間在控制檯應用程序中獲得的標準輸入/輸出。

1:使用sprintf(int sprintf (char * str, const char * format, ...))。它所做的是打印成一個字符串而不是stdout(你不必使用臨時文件)。在此之後,您可以使用MessageBox以及剛剛打印到的字符串。

2:進行一個實際的控制檯窗口(同時保持主一個),並從主窗口到控制檯stdinstdoutstderr重定向。下面是建築類:

ConsoleWindowClass.h:

#pragma once 
#include <windows.h> 
#include <stdio.h> 
#include <fcntl.h> 
#include <io.h> 
#include <iostream> 
#include <fstream> 
class ConsoleWindowClass 
{ 
public: 
    ConsoleWindowClass(void); 
    ~ConsoleWindowClass(void); 
    void Create(); 
}; 

ConsoleWindowClass.cpp:

#include "ConsoleWindowClass.h" 

using namespace std; 

// maximum mumber of lines the output console should have 
static const WORD MAX_CONSOLE_LINES = 500; 

ConsoleWindowClass::ConsoleWindowClass(void) 
{ 
    Create(); 
} 

ConsoleWindowClass::~ConsoleWindowClass(void) 
{ 
} 

void ConsoleWindowClass::Create() 
{ 
    int hConHandle; 
    long lStdHandle; 
    CONSOLE_SCREEN_BUFFER_INFO coninfo; 
    FILE *fp; 

// allocate a console for this app 
    AllocConsole(); 

// set the screen buffer to be big enough to let us scroll text 
    GetConsoleScreenBufferInfo(GetStdHandle(STD_OUTPUT_HANDLE),&coninfo); 
    coninfo.dwSize.Y = MAX_CONSOLE_LINES; 
    SetConsoleScreenBufferSize(GetStdHandle(STD_OUTPUT_HANDLE),coninfo.dwSize); 

// redirect unbuffered STDOUT to the console 
    lStdHandle = (long)GetStdHandle(STD_OUTPUT_HANDLE); 
    hConHandle = _open_osfhandle(lStdHandle, _O_TEXT); 

    fp = _fdopen(hConHandle, "w"); 

    *stdout = *fp; 

    setvbuf(stdout, NULL, _IONBF, 0); 

// redirect unbuffered STDIN to the console 

    lStdHandle = (long)GetStdHandle(STD_INPUT_HANDLE); 
    hConHandle = _open_osfhandle(lStdHandle, _O_TEXT); 

    fp = _fdopen(hConHandle, "r"); 
    *stdin = *fp; 
    setvbuf(stdin, NULL, _IONBF, 0); 

// redirect unbuffered STDERR to the console 
    lStdHandle = (long)GetStdHandle(STD_ERROR_HANDLE); 
    hConHandle = _open_osfhandle(lStdHandle, _O_TEXT); 

    fp = _fdopen(hConHandle, "w"); 

    *stderr = *fp; 

    setvbuf(stderr, NULL, _IONBF, 0); 

// make cout, wcout, cin, wcin, wcerr, cerr, wclog and clog 
// point to console as well 
    ios::sync_with_stdio(); 
} 

在此之後,調用printf()將打印字符串到控制檯。您也可以使用控制檯在其中輸入字符串,並且它們將在主窗口中可用(使用多線程,因此scanf不會暫停您的主程序)。

+0

如果您只想使用C而不是C++,則可以刪除和ios :: sync_with_stdio()。你還必須在沒有課堂的情況下完成。這是最後一行的功能: http://www.cplusplus.com/reference/ios/ios_base/sync_with_stdio/ – Vladivarius

+0

感謝您的解決方案,但請閱讀「編輯」。 – Abhineet

+0

是的,我在寫完答案後才明白這個問題。無論如何,這將有望爲有意尋找重定向的人提供:) – Vladivarius

10

由於您的問題似乎僅供參考,沒有控制檯的Windows應用程序有其stdout,stderr句柄關閉。任何試圖輸出到這些句柄的函數都會被調用,檢查一個打開的句柄,找到它關閉,並返回而不做其他任何事情。

你可能會說,你在這種情況下,輸出最終無處可尋:)

如果你想讀此輸出,那麼你需要通過分配控制檯打開手柄要麼,或使用一個這裏描述的方法。

相關問題