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


112

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

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

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

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


可能的重复这里这里(新的,但用一些有趣的答案AttachConsole从Kernel32.dll中
马克斯

1
@Max,这些问题可能是问题的重复项。在您发布任何一个问题之前2-4年都曾问过这个问题。
罗布(Rob)

Answers:


90

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

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

在下面的类中,您只需要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);
    }
} 

5
可以尝试首先调用AttachConsole(-1)并检查其返回值以附加到父进程的控制台;如果返回false,则调用AllocConsole。但是,该应用程序仍然会先“返回”,然后才将其输出到控制台,如果找到解决方案,我将发布更多内容。另外,如果将WPF应用程序类型设置为“控制台应用程序”,问题会消失,但是如果在启动程序时没有在屏幕上短暂显示控制台,就无法分离控制台,因此它看起来有些尴尬(但是如果可以使用的话) ,效果很好)。
亚历克斯·帕文

2
嗯,实际上不,我认为不可能同时使用这两种方法。控制台应用程序在其PE标头中标记为CUI,因此可以自动与CMD很好地协作。另一方面,GUI应用程序立即将控制权返回给CMD,即使它可以重新连接到控制台,读取和写入操作也会与管道中的下一个输出混合在一起,这显然非常糟糕。另一方面,如果您将应用程序标记为控制台应用程序,则只需要使用CMD即可在应用程序启动时短暂显示。那么你可以使用FreeConsole来分离和附加/后的Alloc等
亚历Paven

1
当Brian的答案同样有效且容易得多时,为什么要这样做。
Wouter Janssens,2012年

2
可能很明显,但是我发现在附加Visual Studio调试器时,Console.WriteLine仍然无法使用该技术。当我在VS之外运行该应用程序时,它很有效。谢谢。
aboy021

2
@Mark是的,但是它不起作用...有一个SetConsoleCtrlHandler功能可以在CTRL_CLOSE_EVENT事件发生时得到通知,但是您无法对其进行任何操作,没有任何东西可以让您的应用程序继续运行。您将被关闭。如果您感觉被黑客入侵,则可以将Windows消息处理程序交换为控制台进程,然后删除WM_CLOSE消息,但我从未尝试过,但是它可以工作。这只是另一个窗口,但话虽这么说,除非您想接受这个想法,否则最好将精力花在做其他事情上。
约翰·莱德格伦

129

右键单击项目的“属性”,“应用程序”选项卡,将“输出类型”更改为“控制台应用程序”,然后它还将具有一个控制台。


2
唯一的问题是您将在后台打开一个cmd,但是它可以:)。
ykatchou 2012年

5
很好,但是当不从cmd.exe执行应用程序时将创建命令行窗口(为一个应用程序创建了两个窗口)。但是为此,还有一个解决方案:您可以通过ShowWindow(hWnd,0)隐藏cmd窗口。stackoverflow.com/a/10416180/1457197。使用此解决方案,仅当从命令行执行WPF应用程序时,您才会在控制台中看到文本。
CoperNick

请注意,在Blend中工作时必须将其切换回“ Window Application”,因为它仅显示“ Console Application”类型的XAML(无权访问“设计视图”)。(截至Blend 2013)

1
不正确的答案。隐藏主窗口。只是控制台出现。
Yash

128

您可以使用

Trace.WriteLine("text");

这将输出到Visual Studio中的“输出”窗口(调试时)。

确保包含诊断程序部件:

using System.Diagnostics;

9
这是最佳答案,但评级却不是最高
kiltek

我同意-这正是op所要求的。替代Console.WriteLine()的绝佳选择-标记为答案的解决方案是一项巧妙的练习,但不合理地包含在生产应用程序中。
nocarrier

4
适用于Windows应用商店(Windows运行时)的PS,相当于Trace.WriteLine是Debug.WriteLine()
nocarrier 2013年

这是一个简单,干净的解决方案,但是对我不起作用。在更新数据库期间,实体框架的种子方法无效。否则,可以在其他地方使用!
Charles W

这是最好的解决方案。如果答案也解释Console.WriteLine为完全不用于WPF应用程序,并且仅用于命令行应用程序,那就更好了。
安德鲁·科斯特

12

尽管约翰·莱德格伦(John Leidegren)一直反对这个想法,但布莱恩(Brian)是正确的。我刚在Visual Studio中运行它。

显然,WPF应用程序默认情况下不会创建“控制台”窗口。

您必须创建WPF应用程序,然后将OutputType更改为“控制台应用程序”。运行项目时,您将看到一个控制台窗口,其前面有WPF窗口。

它看起来不太漂亮,但是我发现它很有用,因为我希望从命令行运行我的应用程序并在其中添加反馈,然后对于某些命令选项,我将显示WPF窗口。


1
完善。做这份工作。
frostymarvelous

10

通过使用命令行重定向,可以看到用于控制台的输出。

例如:

C:\src\bin\Debug\Example.exe > output.txt

将所有内容写入output.txt文件。


最佳答案,因为它很简单并且不需要更改源
弯的

9

旧文章,但是我遇到了这个问题,因此,如果您尝试在Visual Studio的WPF项目中将某些内容输出到Output,则现代方法是:

包括以下内容:

using System.Diagnostics;

然后:

Debug.WriteLine("something");

4

我在输出窗口中使用Console.WriteLine()...


4
自从我第一次看到这个问题以来,这个问题已有4年的历史了。当然,现在问题的措词更好了,我的回答也变得无关紧要了。
erodewald

1

我创建了一个解决方案,混合了varius post的信息。

它是一种表单,其中包含一个标签和一个文本框。控制台输出将重定向到文本框。

还有一个叫做ConsoleView的类,它实现三个public方法:Show(),Close()和Release()。最后一个用于休假,请打开控制台并激活“关闭”按钮以查看结果。

表单称为FrmConsole。这是XAML和c#代码。

使用非常简单:

ConsoleView.Show("Title of the Console");

用于打开控制台。用:

System.Console.WriteLine("The debug message");

用于将文本输出到控制台。

用:

ConsoleView.Close();

对于关闭控制台。

ConsoleView.Release();

离开控制台并启用“关闭”按钮

XAML

<Window x:Class="CustomControls.FrmConsole"
    xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
    xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
    xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
    xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
    xmlns:local="clr-namespace:CustomControls"
    mc:Ignorable="d"
    Height="500" Width="600" WindowStyle="None" ResizeMode="NoResize" WindowStartupLocation="CenterScreen" Topmost="True" Icon="Images/icoConsole.png">
<Grid>
    <Grid.RowDefinitions>
        <RowDefinition Height="40"/>
        <RowDefinition Height="*"/>
        <RowDefinition Height="40"/>
    </Grid.RowDefinitions>
    <Label Grid.Row="0" Name="lblTitulo" HorizontalAlignment="Center" HorizontalContentAlignment="Center" VerticalAlignment="Center" VerticalContentAlignment="Center" FontFamily="Arial" FontSize="14" FontWeight="Bold" Content="Titulo"/>
    <Grid Grid.Row="1">
        <Grid.ColumnDefinitions>
            <ColumnDefinition Width="10"/>
            <ColumnDefinition Width="*"/>
            <ColumnDefinition Width="10"/>
        </Grid.ColumnDefinitions>
        <TextBox Grid.Column="1" Name="txtInner" FontFamily="Arial" FontSize="10" ScrollViewer.CanContentScroll="True" VerticalScrollBarVisibility="Visible" HorizontalScrollBarVisibility="Visible" TextWrapping="Wrap"/>
    </Grid>
    <Button Name="btnCerrar" Grid.Row="2" Content="Cerrar" Width="100" Height="30" HorizontalAlignment="Center" HorizontalContentAlignment="Center" VerticalAlignment="Center" VerticalContentAlignment="Center"/>
</Grid>

窗口的代码:

partial class FrmConsole : Window
{
    private class ControlWriter : TextWriter
    {
        private TextBox textbox;
        public ControlWriter(TextBox textbox)
        {
            this.textbox = textbox;
        }

        public override void WriteLine(char value)
        {
            textbox.Dispatcher.Invoke(new Action(() =>
            {
                textbox.AppendText(value.ToString());
                textbox.AppendText(Environment.NewLine);
                textbox.ScrollToEnd();
            }));
        }

        public override void WriteLine(string value)
        {
            textbox.Dispatcher.Invoke(new Action(() =>
            {
                textbox.AppendText(value);
                textbox.AppendText(Environment.NewLine);
                textbox.ScrollToEnd();
            }));
        }

        public override void Write(char value)
        {
            textbox.Dispatcher.Invoke(new Action(() =>
            {
                textbox.AppendText(value.ToString());
                textbox.ScrollToEnd();
            }));
        }

        public override void Write(string value)
        {
            textbox.Dispatcher.Invoke(new Action(() =>
            {
                textbox.AppendText(value);
                textbox.ScrollToEnd();
            }));
        }

        public override Encoding Encoding
        {
            get { return Encoding.UTF8; }

        }
    }

    //DEFINICIONES DE LA CLASE
    #region DEFINICIONES DE LA CLASE

    #endregion


    //CONSTRUCTORES DE LA CLASE
    #region CONSTRUCTORES DE LA CLASE

    public FrmConsole(string titulo)
    {
        InitializeComponent();
        lblTitulo.Content = titulo;
        Clear();
        btnCerrar.Click += new RoutedEventHandler(BtnCerrar_Click);
        Console.SetOut(new ControlWriter(txtInner));
        DesactivarCerrar();
    }

    #endregion


    //PROPIEDADES
    #region PROPIEDADES

    #endregion


    //DELEGADOS
    #region DELEGADOS

    private void BtnCerrar_Click(object sender, RoutedEventArgs e)
    {
        Close();
    }

    #endregion


    //METODOS Y FUNCIONES
    #region METODOS Y FUNCIONES

    public void ActivarCerrar()
    {
        btnCerrar.IsEnabled = true;
    }

    public void Clear()
    {
        txtInner.Clear();
    }

    public void DesactivarCerrar()
    {
        btnCerrar.IsEnabled = false;
    }

    #endregion  
}

ConsoleView类的代码

static public class ConsoleView
{
    //DEFINICIONES DE LA CLASE
    #region DEFINICIONES DE LA CLASE
    static FrmConsole console;
    static Thread StatusThread;
    static bool isActive = false;
    #endregion

    //CONSTRUCTORES DE LA CLASE
    #region CONSTRUCTORES DE LA CLASE

    #endregion

    //PROPIEDADES
    #region PROPIEDADES

    #endregion

    //DELEGADOS
    #region DELEGADOS

    #endregion

    //METODOS Y FUNCIONES
    #region METODOS Y FUNCIONES

    public static void Show(string label)
    {
        if (isActive)
        {
            return;
        }

        isActive = true;
        //create the thread with its ThreadStart method
        StatusThread = new Thread(() =>
        {
            try
            {
                console = new FrmConsole(label);
                console.ShowDialog();
                //this call is needed so the thread remains open until the dispatcher is closed
                Dispatcher.Run();
            }
            catch (Exception)
            {
            }
        });

        //run the thread in STA mode to make it work correctly
        StatusThread.SetApartmentState(ApartmentState.STA);
        StatusThread.Priority = ThreadPriority.Normal;
        StatusThread.Start();

    }

    public static void Close()
    {
        isActive = false;
        if (console != null)
        {
            //need to use the dispatcher to call the Close method, because the window is created in another thread, and this method is called by the main thread
            console.Dispatcher.InvokeShutdown();
            console = null;
            StatusThread = null;
        }

        console = null;
    }

    public static void Release()
    {
        isActive = false;
        if (console != null)
        {
            console.Dispatcher.Invoke(console.ActivarCerrar);
        }

    }
    #endregion
}

我希望这个结果有用。



-17

据我所知,Console.WriteLine()仅适用于控制台应用程序。我认为这是您的问题。


1
我不知道WPF,但是WinForms肯定不是这种情况。Console.WriteLine在那里可以正常工作,但是,当然,您不会看到控制台,您会在Debugger输出窗口中看到它,并且如果您收听标准输出。
杰夫·耶茨

2
您可以将项目设置为控制台应用程序,它仍将作为Windows应用程序运行,但也将具有可见的控制台
Mark Cidade

这是不正确的,非控制台应用程序的构建过程不会通过授权附加控制台。您可以在调用Console.Write之前先调用AllocConsole()Win32 API函数来手动执行此操作。然后,将初始化Console类以使其与该Console窗口一起使用。
约翰·莱德格伦
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.