2012-10-24 75 views
0

我有一个GUI C#应用程序,它使用用AllocConsole创建的控制台。它在正常情况下工作,但是当应用程序在Visual Studio下以调试模式启动时,所有输出都将在Visual Studio输出窗口中结束。我如何阻止?如何阻止VC#将控制台输出重定向到输出窗口?

我正在使用C#3.5和Visual Studio Pro 2010. 进程托管选项已关闭。

+0

不,这应该工作得很好,如果C#代码*实际*使用Console.Write/Line()。如果它执行诸如Debug.Print或跟踪之类的操作,则输出由OutputDebugString()写入,并最终在调试器窗口中结束。输出窗口。考虑SysInternals的DebugView工具。 –

+0

我使用Console.WriteLine,它结束于Visual Studio输出窗口,而不是我的控制台。 –

+0

很难解释。尝试关闭托管过程选项。 –

回答

3

我的解决方案是将标准流重置为新创建的控制台。这里是代码:

public static class ConsoleHelper 
{ 
    /// <summary> 
    /// Allocates a console and resets the standard stream handles. 
    /// </summary> 
    public static void Alloc() 
    { 
     if (!AllocConsole()) 
      throw new Win32Exception(); 
     SetStdHandle(StdHandle.Output, GetConsoleStandardOutput()); 
     SetStdHandle(StdHandle.Input, GetConsoleStandardInput()); 
    } 

    private static IntPtr GetConsoleStandardInput() 
    { 
     var handle = CreateFile 
      ("CONIN$" 
      , DesiredAccess.GenericRead | DesiredAccess.GenericWrite 
      , FileShare.ReadWrite 
      , IntPtr.Zero 
      , FileMode.Open 
      , FileAttributes.Normal 
      , IntPtr.Zero 
      ); 
     if (handle == InvalidHandleValue) 
      throw new Win32Exception(); 
     return handle; 
    } 

    private static IntPtr GetConsoleStandardOutput() 
    { 
     var handle = CreateFile 
      ("CONOUT$" 
      , DesiredAccess.GenericWrite | DesiredAccess.GenericWrite 
      , FileShare.ReadWrite 
      , IntPtr.Zero 
      , FileMode.Open 
      , FileAttributes.Normal 
      , IntPtr.Zero 
      ); 
     if (handle == InvalidHandleValue) 
      throw new Win32Exception(); 
     return handle; 
    } 

    [DllImport("kernel32.dll", SetLastError = true)] 
    private static extern bool AllocConsole(); 

    [DllImport("kernel32.dll")] 
    private static extern bool SetStdHandle(StdHandle nStdHandle, IntPtr hHandle); 

    [DllImport("kernel32.dll", SetLastError = true, CharSet = CharSet.Auto)] 
    private static extern IntPtr CreateFile 
     (        string   lpFileName 
     , [MarshalAs(UnmanagedType.U4)] DesiredAccess dwDesiredAccess 
     , [MarshalAs(UnmanagedType.U4)] FileShare  dwShareMode 
     ,        IntPtr   lpSecurityAttributes 
     , [MarshalAs(UnmanagedType.U4)] FileMode  dwCreationDisposition 
     , [MarshalAs(UnmanagedType.U4)] FileAttributes dwFlagsAndAttributes 
     ,        IntPtr   hTemplateFile 
     ); 

    [Flags] 
    enum DesiredAccess : uint 
    { 
     GenericRead = 0x80000000, 
     GenericWrite = 0x40000000, 
     GenericExecute = 0x20000000, 
     GenericAll  = 0x10000000 
    } 

    private enum StdHandle : int 
    { 
     Input = -10, 
     Output = -11, 
     Error = -12 
    } 

    private static readonly IntPtr InvalidHandleValue = new IntPtr(-1); 
} 
+0

哪个命名空间做DesiredAccess来自? –

+0

啊,好像是从http://www.pinvoke.net/default.aspx/kernel32.createfile派生的 –

0

您可以通过启动该应用程序不同,像这样(参考文献1)输出重定向:

// 
// Setup the process with the ProcessStartInfo class. 
// 
ProcessStartInfo start = new ProcessStartInfo(); 
start.FileName = @"C:\MyExe.exe"; // Specify exe name. 
start.UseShellExecute = false; 
start.RedirectStandardOutput = true; 

也可以消耗的输出(如果需要的话)由以下(参考文献1)

// 
// Start the process. 
// 
using (Process process = Process.Start(start)) 
{ 
    // 
    // Read in all the text from the process with the StreamReader. 
    // 
    using (StreamReader reader = process.StandardOutput) 
    { 
    string result = reader.ReadToEnd(); 
    Console.Write(result); 
    } 
} 

参考文献: (1)http://www.dotnetperls.com/redirectstandardoutput

+0

我没有重定向另一个可执行文件的流。 –

相关问题