2017-01-12 17 views
2

OK、私はいつでもWinformが必要になると、コンソール出力にはAllocConsole()メソッドを使用しました。それはコンソールに書き込むことになる。Console.Out出力がAllocConsole()で必要な出力ウィンドウに表示されています

VS 2015以下を使用すると、デバッグモードのAllocConsoleは常に正常に動作し、Console.WriteLineが正しく書き込まれました。 VS 2017を使用すると、コンソールはAllocConsoleが呼び出されたときに表示されますが、console.WriteLine出力はコンソールに行くのではなく、Visual Studioの出力ウィンドウに移動します。

私は色に大きく依存しているので、出力ウィンドウではなくAllocConsoleを使用することをお勧めします。私はこれを修正する方法についてたくさんの検索をしましたが、私は答えを見つけることができません。

+1

VS2017は依然としてRC段階にあります。[私は直接質問します](https://docs.microsoft.com/en-us/visualstudio/ide/how-to-reporta-a-問題視覚スタジオ2017) –

+0

可能な重複...これはプロジェクトで設定できるものです。 Chazの答えをチェックする[フォームにコンソール出力/ウィンドウを表示するにはどうすればいいですか?](http://stackoverflow.com/questions/4362111/how-do-i-show-a-console-output-window-in-a -forms-application) – JohnG

+0

@JohnGありがとう!必要なものを正確に行うためのもっと簡単な方法。 – apotter96

答えて

2

AllocConsole()はVS 2017が "debug stdout redirect magic"を実行しているため、それ自体では機能しません。これを修正するには、AllocConsole()でコンソールを作成し、stdoutハンドルを修正する必要があります。ここで

は、私が見つけた切り取られる:

[DllImport("kernel32.dll", 
    EntryPoint = "AllocConsole", 
    SetLastError = true, 
    CharSet = CharSet.Auto, 
    CallingConvention = CallingConvention.StdCall)] 
private static extern int AllocConsole(); 

[DllImport("kernel32.dll", SetLastError = true)] 
private static extern IntPtr CreateFile(
    string lpFileName, 
    uint dwDesiredAccess, 
    uint dwShareMode, 
    uint lpSecurityAttributes, 
    uint dwCreationDisposition, 
    uint dwFlagsAndAttributes, 
    uint hTemplateFile); 

private const int MY_CODE_PAGE = 437; 
private const uint GENERIC_WRITE = 0x40000000; 
private const uint FILE_SHARE_WRITE = 0x2; 
private const uint OPEN_EXISTING = 0x3; 

public static void CreateConsole() 
{ 
    AllocConsole(); 

    IntPtr stdHandle = CreateFile(
     "CONOUT$", 
     GENERIC_WRITE, 
     FILE_SHARE_WRITE, 
     0, OPEN_EXISTING, 0, 0 
    ); 

    SafeFileHandle safeFileHandle = new SafeFileHandle(stdHandle, true); 
    FileStream fileStream = new FileStream(safeFileHandle, FileAccess.Write); 
    Encoding encoding = System.Text.Encoding.GetEncoding(MY_CODE_PAGE); 
    StreamWriter standardOutput = new StreamWriter(fileStream, encoding); 
    standardOutput.AutoFlush = true; 
    Console.SetOut(standardOutput); 

    Console.Write("This will show up in the Console window."); 
} 

感謝Ramkumarラメシュにする回避策のために: Console Output is gone in VS2017

1

ビルwischiからの回答に、あなたがのためのプロパティをしたい場合例えばConsole.ForegroundColorのように、正しく動作するためには、static Consoleクラスを使用する必要があります。desiredAccessをGENERIC_READ | GENERIC_WRITE。理由はこれはコンソールが内部的にGetConsoleScreenBufferInfoを使用していることと、GENERIC_WRITEのみでCreateFileから返されたハンドルでそのメソッドを使用しようとしたときにACCESS_DENIEDエラーが発生したためです。

[DllImport("kernel32.dll")] 
private static extern bool AllocConsole(); 

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

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

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

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

public static void CreateConsole() 
{ 
    if (AllocConsole()) 
    { 
     //https://developercommunity.visualstudio.com/content/problem/12166/console-output-is-gone-in-vs2017-works-fine-when-d.html 
     // Console.OpenStandardOutput eventually calls into GetStdHandle. As per MSDN documentation of GetStdHandle: http://msdn.microsoft.com/en-us/library/windows/desktop/ms683231(v=vs.85).aspx will return the redirected handle and not the allocated console: 
     // "The standard handles of a process may be redirected by a call to SetStdHandle, in which case GetStdHandle returns the redirected handle. If the standard handles have been redirected, you can specify the CONIN$ value in a call to the CreateFile function to get a handle to a console's input buffer. Similarly, you can specify the CONOUT$ value to get a handle to a console's active screen buffer." 
     // Get the handle to CONOUT$.  
     var stdOutHandle = CreateFile("CONOUT$", DesiredAccess.GenericRead | DesiredAccess.GenericWrite, FileShare.ReadWrite, 0, FileMode.Open, FileAttributes.Normal, 0); 

     if (stdOutHandle == new IntPtr(-1)) 
     { 
      throw new Win32Exception(Marshal.GetLastWin32Error()); 
     } 

     if (!SetStdHandle(StdHandle.Output, stdOutHandle)) 
     { 
      throw new Win32Exception(Marshal.GetLastWin32Error()); 
     } 

     var standardOutput = new StreamWriter(Console.OpenStandardOutput()); 
     standardOutput.AutoFlush = true; 
     Console.SetOut(standardOutput); 
    } 
} 
関連する問題