非常に単純なWPFテストアプリケーションからConsole.WriteLine()を使用していますが、コマンドラインからアプリケーションを実行すると、コンソールに何も書き込まれません。誰がここで何が起こっているのか知っていますか?
VS 2008でWPFアプリケーションを作成し、Console.WriteLine( "text")を実行する場所に追加するだけで再現できます。何か案は?
今必要なのは、Console.WriteLine()のような単純なものです。 log4netまたはその他のロギングソリューションを使用できることはわかっていますが、このアプリケーションにはそれほど多くの機能は必要ありません。
編集: Console.WriteLine()はコンソールアプリケーション用であることを思い出すべきでした。まあ、愚かな質問はありませんか? :-)ここでは、System.Diagnostics.Trace.WriteLine()とDebugViewを使用します。
実際にConsole.Writeメソッドを呼び出す前に、コンソールウィンドウを手動で作成する必要があります。これにより、プロジェクトの種類を変更せずにコンソールが正しく動作するようになります(WPFアプリケーションの場合は動作しません)。
以下に、ConsoleManagerクラスがどのように見えるか、およびプロジェクトタイプに関係なくConsoleを有効/無効にするために使用できる方法の完全なソースコードの例を示します。
次のクラスでは、_Console.Write
_...を呼び出す前に、どこかにConsoleManager.Show()
を記述するだけです。
_[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);
}
}
_
プロジェクト、[プロパティ]、[アプリケーション]タブを右クリックし、[出力タイプ]を[コンソールアプリケーション]に変更すると、コンソールも表示されます。
使用できます
Trace.WriteLine("text");
これにより、Visual Studioの「出力」ウィンドウに出力されます(デバッグ時)。
診断アセンブリが含まれていることを確認してください。
using System.Diagnostics;
ジョン・ライデグレンはアイデアを打ち続けていますが、ブライアンは正しいです。 Visual Studioで動作するようになりました。
WPFアプリケーションをクリアしても、デフォルトではコンソールウィンドウは作成されません。
WPFアプリケーションを作成してから、OutputTypeを「コンソールアプリケーション」に変更する必要があります。プロジェクトを実行すると、WPFウィンドウが前面にあるコンソールウィンドウが表示されます。
見た目はあまりきれいではありませんが、コマンドラインからアプリを実行してフィードバックを送り、特定のコマンドオプションについてはWPFウィンドウを表示するので便利です。
コマンドラインリダイレクト を使用すると、コンソール向けの出力を表示できます。
例えば:
C:\src\bin\Debug\Example.exe > output.txt
すべてのコンテンツをoutput.txt
ファイルに書き込みます。
古い投稿ですが、Visual StudioのWPFプロジェクトのOutputに何かを出力しようとしている場合、これに遭遇しました。現代的な方法は次のとおりです。
これを含める:
using System.Diagnostics;
その後:
Debug.WriteLine("something");
出力ウィンドウで使用するためにConsole.WriteLine()を使用します...
さまざまな投稿の情報を混合したソリューションを作成しました。
ラベルと1つのテキストボックスを含むフォーム。コンソール出力はテキストボックスにリダイレクトされます。
3つのパブリックメソッドを実装するConsoleViewというクラスもあります:Show()、Close()、Release()。最後の1つは、コンソールを開いたままにして、結果を表示するために[閉じる]ボタンをアクティブにするためのものです。
フォームは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
}
この結果が役に立つことを願っています。
この投稿をチェックして、自分にとって非常に役に立ちました。コードサンプルをダウンロードします。
http://www.codeproject.com/Articles/335909/Embedding-a-Console-in-a-C-Application