我有一个GUI C#应用程序,它使用用AllocConsole
创建的控制台。它在正常情况下工作,但是当应用程序在Visual Studio下以调试模式启动时,所有输出都将在Visual Studio输出窗口中结束。我如何阻止?如何阻止VC#将控制台输出重定向到输出窗口?
我正在使用C#3.5和Visual Studio Pro 2010. 进程托管选项已关闭。
我有一个GUI C#应用程序,它使用用AllocConsole
创建的控制台。它在正常情况下工作,但是当应用程序在Visual Studio下以调试模式启动时,所有输出都将在Visual Studio输出窗口中结束。我如何阻止?如何阻止VC#将控制台输出重定向到输出窗口?
我正在使用C#3.5和Visual Studio Pro 2010. 进程托管选项已关闭。
我的解决方案是将标准流重置为新创建的控制台。这里是代码:
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);
}
哪个命名空间做DesiredAccess来自? –
啊,好像是从http://www.pinvoke.net/default.aspx/kernel32.createfile派生的 –
您可以通过启动该应用程序不同,像这样(参考文献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);
}
}
我没有重定向另一个可执行文件的流。 –
不,这应该工作得很好,如果C#代码*实际*使用Console.Write/Line()。如果它执行诸如Debug.Print或跟踪之类的操作,则输出由OutputDebugString()写入,并最终在调试器窗口中结束。输出窗口。考虑SysInternals的DebugView工具。 –
我使用Console.WriteLine,它结束于Visual Studio输出窗口,而不是我的控制台。 –
很难解释。尝试关闭托管过程选项。 –