将标准输出重定向回控制台

Redirect stdout back to console

有很多关于将 stdout 和 stderr 重定向到文件而不是控制台的文档。你如何将它重新重定向回来?下面的代码显示了我的意图,但只输出一次 "stdout is printed to console"。

我猜我需要获取控制台输出缓冲区,将其存储在某处,将标准输出重定向到文件,然后恢复控制台缓冲区?

#pragma warning(disable:4996)

#include <cstdio>

int main()
{
    std::printf("stdout is printed to console\n");

    if (std::freopen("redir.txt", "w", stdout)) {
        std::printf("stdout is redirected to a file\n"); // this is written to redir.txt
        std::fclose(stdout);

        std::printf("stdout is printed to console\n");
    }

    getchar();
    return 0;
}

感谢上面评论中的文章,我找到了我需要的信息。 dup 和 dup2 函数正是我所需要的。请注意,基于信息 here dup and dup2 are deprecated in favour or _dup and _dup2. A working example can be found on MSDN here,但在下面重复以防 link 将来中断。

// crt_dup.c
// This program uses the variable old to save
// the original stdout. It then opens a new file named
// DataFile and forces stdout to refer to it. Finally, it
// restores stdout to its original state.

#include <io.h>
#include <stdlib.h>
#include <stdio.h>

int main( void )
{
   int old;
   FILE *DataFile;

   old = _dup( 1 );   // "old" now refers to "stdout"
                      // Note:  file descriptor 1 == "stdout"
   if( old == -1 )
   {
      perror( "_dup( 1 ) failure" );
      exit( 1 );
   }
   _write( old, "This goes to stdout first\n", 26 );
   if( fopen_s( &DataFile, "data", "w" ) != 0 )
   {
      puts( "Can't open file 'data'\n" );
      exit( 1 );
   }

   // stdout now refers to file "data"
   if( -1 == _dup2( _fileno( DataFile ), 1 ) )
   {
      perror( "Can't _dup2 stdout" );
      exit( 1 );
   }
   puts( "This goes to file 'data'\n" );

   // Flush stdout stream buffer so it goes to correct file
   fflush( stdout );
   fclose( DataFile );

   // Restore original stdout
   _dup2( old, 1 );
   puts( "This goes to stdout\n" );
   puts( "The file 'data' contains:" );
   _flushall();
   system( "type data" );
}