131

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

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

我现在需要的只是简单的Console.WriteLine(). 我意识到我可以使用 log4net 或其他一些日志记录解决方案,但我真的不需要这个应用程序的那么多功能。

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

4

10 回答 10

181

您可以使用

Trace.WriteLine("text");

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

确保包含诊断程序集:

using System.Diagnostics;
于 2008-10-02T02:20:35.100 回答
141

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

于 2008-10-02T02:17:12.097 回答
96

在实际调用任何 Console.Write 方法之前,您必须手动创建一个控制台窗口。这将使控制台在不更改项目类型的情况下正常工作(对于 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);
    }
} 
于 2009-04-05T07:14:31.997 回答
20

旧帖子,但我遇到了这个,所以如果你想在 Visual Studio 的 WPF 项目中输出一些东西到输出,现代方法是:

包括这个:

using System.Diagnostics;

进而:

Debug.WriteLine("something");
于 2017-01-19T02:07:11.710 回答
15

尽管 John Leidegren 不断否定这个想法,但 Brian 是正确的。我刚刚让它在 Visual Studio 中工作。

需要明确的是,WPF 应用程序默认情况下不会创建控制台窗口。

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

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

于 2009-04-27T11:52:01.743 回答
12

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

例如:

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

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

于 2014-05-30T15:10:06.950 回答
4

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

于 2008-10-02T05:50:56.793 回答
1

我创建了一个解决方案,混合了各种帖子的信息。

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

还有一个名为 ConsoleView 的类实现了三个公共方法: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
}

我希望这个结果有用。

于 2018-08-06T06:42:03.037 回答
0

看看这个帖子,对自己很有帮助。下载代码示例:

http://www.codeproject.com/Articles/335909/Embedding-a-Console-in-aC-Application

于 2014-08-12T03:08:27.217 回答
0

我最近也遇到了这个问题。

如果 Brians 解决方案(在项目设置中更改为控制台应用程序)没有解决问题:尝试将目标框架从 .NET 5.0 更改为 .NET Core 3.1。

于 2021-10-26T12:51:02.597 回答