我正在编写一个可以作为标准WinForms应用程序启动的应用程序,也可以从命令行以无人参与模式启动。该应用程序是使用VS 2k5标准WinForms模板构建的。
当从命令行执行应用程序时,我希望它输出可由执行应用程序的脚本捕获的信息。当我直接从Console.WriteLine()执行此操作时,输出不会出现,但可以通过管道传输到文件来捕获。
另一方面,我可以通过在kernel32上对AllocConsole()执行P / Invoke来强制应用程序弹出第二个控制台。但这不是我想要的。我希望输出出现在调用应用程序的同一窗口中。
这是允许我从命令行弹出控制台的显着代码:
<STAThread()> Public Shared Sub Main()
If My.Application.CommandLineArgs.Count = 0 Then
Dim frm As New ISECMMParamUtilForm()
frm.ShowDialog()
Else
Try
ConsoleControl.AllocConsole()
Dim exMan As New UnattendedExecutionManager(ConvertArgs())
IsInConsoleMode = True
OutputMessage("Application started.")
If Not exMan.SetSettings() Then
OutputMessage("Execution failed.")
End If
Catch ex As Exception
Console.WriteLine(ex.ToString())
Finally
ConsoleControl.FreeConsole()
End Try
End If
End Sub
Public Shared Sub OutputMessage(ByVal msg As String, Optional ByVal isError As Boolean = False)
Trace.WriteLine(msg)
If IsInConsoleMode Then
Console.WriteLine(msg)
End If
If isError Then
EventLog.WriteEntry("ISE CMM Param Util", msg, EventLogEntryType.Error)
Else
EventLog.WriteEntry("ISE CMM Param Util", msg, EventLogEntryType.Information)
End If
End Sub
答案 0 :(得分:9)
Raymond Chen最近发布了一篇关于此问题的简短文章(问题发布于SO之后一个月):
How do I write a program that can be run either as a console or a GUI application?
你不能,但你可以试着伪造它。
每个PE应用程序都包含一个字段 在其标题中指定哪个 它被设计为运行的子系统 下。你可以说
IMAGE_SUBSYSTEM_WINDOWS_GUI
要标记 你自己作为Windows GUI应用程序, 或者你可以说IMAGE_SUBSYSTEM_WINDOWS_CUI
说 你是一个控制台应用程序。如果 你是GUI应用程序,然后是 程序将在没有控制台的情况下运行。子系统确定如何 内核准备执行 该计划的环境。如果 程序被标记为在运行中 控制台子系统,然后是内核 将程序的控制台连接到 其父级的控制台,创建一个 如果父级没有,则为新控制台 一个控制台。 (这是不完整的 描述,但细节不是 与讨论有关。)关于 另一方面,如果程序被标记 然后,作为GUI应用程序运行 内核将运行该程序 没有任何控制台。
在那篇文章中,他指出了张俊峰的另一篇文章,讨论了几个程序(Visual Studio和ildasm)如何实现这种行为:
How to make an application as both GUI and Console application?
在VisualStudio案例中,实际上有两个二进制文件:devenv.com和devenv.exe。 Devenv.com是一款控制台应用。 Devenv.exe是一个GUI应用程序。当您键入devenv时,由于Win32探测规则,会执行devenv.com。如果没有输入,devenv.com将启动devenv.exe,并退出。如果有输入,devenv.com会将它们作为普通的控制台应用程序处理。
在ildasm案例中,只有一个二进制文件:ildasm.exe。它首先被编译为GUI应用程序。稍后editbin.exe用于将其标记为控制台子系统。在其主要方法中,它确定是否需要以控制台模式或GUI模式运行。如果需要以GUI模式运行,它会将自身重新启动为GUI应用程序。
在对Raymond Chen的文章的评论中,laonianren将此添加到Junfeng Zhang关于Visual Studio如何工作的简要描述中:
devenv.com是一个通用的控制台模式存根应用程序。当它运行时,它会创建三个管道来重定向控制台的stdin,stdout和stderr。然后它找到自己的名字(通常是devenv.com),用“.exe”替换“.com”并使用stdin管道的读取端和stdout的写入端启动新的应用程序(即devenv.exe)和stderr管道作为标准手柄。然后它就坐下来等待devenv.exe退出并在控制台和管道之间复制数据。
因此即使devenv.exe是一个gui应用程序,它也可以使用标准句柄读写“父”控制台。
您可以将devenv.com自己用于myapp.exe,方法是将其重命名为myapp.com。但你不能在实践中因为它属于MS。
答案 1 :(得分:2)
更新1:
如迈克尔·伯尔的回答,Raymond Chen最近posted a short article about this。我很高兴看到我的猜测并非完全错误。
更新0:
免责声明:这个“答案”主要是猜测。我发布它只是因为已经有足够的时间来确定没有多少人能够得到看起来像一个基本问题的答案。
我认为应用程序是gui或console的“决定”是在编译时而不是在运行时进行的。因此,如果您将应用程序编译为gui应用程序,即使您没有显示gui,它仍然是一个gui应用程序,并且没有控制台。如果您选择将其编译为控制台应用程序,那么至少在移动到gui“模式”之前,控制台窗口会闪烁。我不知道托管代码是否可行。
问题是根本的,我认为,因为控制台应用程序必须“控制”调用控制台应用程序。并且必须在子应用程序的代码运行之前执行此操作。