小编典典

WPF应用程序没有输出到控制台吗?

c#

我从一个非常简单的WPF测试应用程序中使用Console.WriteLine(),但是当我从命令行执行该应用程序时,却看不到任何内容写入控制台。有人知道这里可能会发生什么吗?

我可以通过在VS 2008中创建WPF应用程序并在要执行的任何位置简单地添加Console.WriteLine(“ text”)来重现它。有任何想法吗?

我现在所需要的只是与Console.WriteLine()一样简单的东西。我意识到我可以使用log4net或其他日志记录解决方案,但是我确实不需要此应用程序那么多的功能。

编辑:
我应该记得Console.WriteLine()是用于控制台应用程序的。哦,没有愚蠢的问题,对吧?:-)我现在只使用System.Diagnostics.Trace.WriteLine()和DebugView。


阅读 988

收藏
2020-05-19

共1个答案

小编典典

在实际调用任何Console.Write方法之前,您必须手动创建一个Console窗口。这将使控制台正常运行,而无需更改项目类型(WPF应用程序将无法使用)。

这是一个完整的源代码示例,它说明ConsoleManager类的外观,以及如何使用它来启用/禁用控制台,而与项目类型无关。

在下面的类中,您只需要ConsoleManager.Show()在调用Console.Write

[SuppressUnmanagedCodeSecurity]
public static class ConsoleManager
{
    private const string Kernel32_DllName = "kernel32.dll";

    [DllImport(Kernel32_DllName)]
    private static extern bool AllocConsole();

    [DllImport(Kernel32_DllName)]
    private static extern bool FreeConsole();

    [DllImport(Kernel32_DllName)]
    private static extern IntPtr GetConsoleWindow();

    [DllImport(Kernel32_DllName)]
    private static extern int GetConsoleOutputCP();

    public static bool HasConsole
    {
        get { return GetConsoleWindow() != IntPtr.Zero; }
    }

    /// <summary>
    /// Creates a new console instance if the process is not attached to a console already.
    /// </summary>
    public static void Show()
    {
        //#if DEBUG
        if (!HasConsole)
        {
            AllocConsole();
            InvalidateOutAndError();
        }
        //#endif
    }

    /// <summary>
    /// If the process has a console attached to it, it will be detached and no longer visible. Writing to the System.Console is still possible, but no output will be shown.
    /// </summary>
    public static void Hide()
    {
        //#if DEBUG
        if (HasConsole)
        {
            SetOutAndErrorNull();
            FreeConsole();
        }
        //#endif
    }

    public static void Toggle()
    {
        if (HasConsole)
        {
            Hide();
        }
        else
        {
            Show();
        }
    }

    static void InvalidateOutAndError()
    {
        Type type = typeof(System.Console);

        System.Reflection.FieldInfo _out = type.GetField("_out",
            System.Reflection.BindingFlags.Static | System.Reflection.BindingFlags.NonPublic);

        System.Reflection.FieldInfo _error = type.GetField("_error",
            System.Reflection.BindingFlags.Static | System.Reflection.BindingFlags.NonPublic);

        System.Reflection.MethodInfo _InitializeStdOutError = type.GetMethod("InitializeStdOutError",
            System.Reflection.BindingFlags.Static | System.Reflection.BindingFlags.NonPublic);

        Debug.Assert(_out != null);
        Debug.Assert(_error != null);

        Debug.Assert(_InitializeStdOutError != null);

        _out.SetValue(null, null);
        _error.SetValue(null, null);

        _InitializeStdOutError.Invoke(null, new object[] { true });
    }

    static void SetOutAndErrorNull()
    {
        Console.SetOut(TextWriter.Null);
        Console.SetError(TextWriter.Null);
    }
}
2020-05-19