Delphi 控制台应用程序可以从现有控制台窗口的命令行运行,并且可以通过双击其图标来运行。在后一种情况下,它将创建自己的控制台窗口,并在应用程序终止后关闭它。
如何判断我的控制台应用程序是否创建了自己的窗口?
我想检测到这一点,以便我可以显示一条消息,如“按 Enter 关闭窗口”,让用户在窗口关闭之前阅读显示的内容。显然,如果应用程序是从命令行运行的,那么这样做是不合适的。
我正在使用 Delphi 2010,以防万一。
Delphi 控制台应用程序可以从现有控制台窗口的命令行运行,并且可以通过双击其图标来运行。在后一种情况下,它将创建自己的控制台窗口,并在应用程序终止后关闭它。
如何判断我的控制台应用程序是否创建了自己的窗口?
我想检测到这一点,以便我可以显示一条消息,如“按 Enter 关闭窗口”,让用户在窗口关闭之前阅读显示的内容。显然,如果应用程序是从命令行运行的,那么这样做是不合适的。
我正在使用 Delphi 2010,以防万一。
您基本上有两件事要测试:
应用程序控制台是否在进程之间共享?如果您用于cmd.exe
运行控制台应用程序,它将默认共享控制台,因此您不需要显示“按 Enter 关闭窗口”消息。
输出是否重定向到文件?如果是这样,也没有必要显示该消息。
对于第一个,有一个简单的解决方案,形式为GetConsoleProcessList()
Windows API 函数。不幸的是,它仅适用于 Windows XP 和更高版本,但也许这对您来说已经足够了。它不在 Delphi 2009Windows
单元中,因此您必须自己导入它:
function GetConsoleProcessList(lpdwProcessList: PDWORD;
dwProcessCount: DWORD): DWORD; stdcall; external 'kernel32.dll';
当然,如果您的软件能够在早期的 Windows 版本上运行,您应该使用LoadLibrary()
andGetProcAddress()
代替。
由于您只对进程句柄的数量是否高于 1 感兴趣,因此您可以使用非常小的句柄缓冲区来调用它,例如:
var
HandleCount: DWORD;
ProcessHandle: DWORD;
begin
HandleCount := GetConsoleProcessList(@ProcessHandle, 1);
// ...
end;
如果您的句柄计数大于 1,则您有其他进程保持控制台打开,因此您可以跳过显示消息。
您可以使用GetFileInformationByHandle()
Windows API 函数来检查您的控制台输出句柄是否引用了真实文件:
var
StdOutHandle: THandle;
IsNotRedirected: boolean;
FileInfo: TByHandleFileInformation;
begin
StdOutHandle := GetStdHandle(STD_OUTPUT_HANDLE);
IsNotRedirected := not GetFileInformationByHandle(StdOutHandle, FileInfo)
and (GetLastError = ERROR_INVALID_HANDLE);
// ...
end;
这段代码只是为了让你开始,我敢肯定有一些极端情况没有得到正确处理。
我过去使用过类似下面的东西:
program ConsoleTest;
{$APPTYPE CONSOLE}
uses Windows;
function GetConsoleWindow: HWND; stdcall; external kernel32 name 'GetConsoleWindow';
function IsOwnConsoleWindow: Boolean;
//ONLY POSSIBLE FOR CONSOLE APPS!!!
//If False, we're being called from the console;
//If True, we have our own console (we weren't called from console)
var pPID: DWORD;
begin
GetWindowThreadProcessId (GetConsoleWindow,pPID);
Result:= (pPID = GetCurrentProcessId);
end;
begin
writeln ('Hello ');
if IsOwnConsoleWindow then begin
writeln ('Press enter to close console');
readln;
end;
end.
我使用(不记得我在哪里找到它):
function WasRanFromConsole() : Boolean;
var
SI: TStartupInfo;
begin
SI.cb := SizeOf(TStartupInfo);
GetStartupInfo(SI);
Result := ((SI.dwFlags and STARTF_USESHOWWINDOW) = 0);
end;
然后像这样使用它:
if (not WasRanFromConsole()) then
begin
Writeln('');
Writeln('Press ENTER to continue');
Readln;
end;
我知道,这是一个旧线程,但我有一个很好的解决方案。
您不必弄乱批处理文件。诀窍在于 exe 的类型,它的子系统属性。将 exe 编译为 GUI 应用程序后(没有 {$APPTYPE CONSOLE} 指令,您必须将其子系统属性 IMAGE_SUBSYSTEM_WINDOWS_GUI 更改为 IMAGE_SUBSYSTEM_WINDOWS_CUI。好消息是当您从控制台执行控制台应用程序时,它不会显示额外的控制台窗口,并且在那时你不需要像“按 Enter 关闭窗口”这样的消息。编辑:如果你在控制台应用程序中启动另一个控制台应用程序,就像我在我的项目中所做的那样)
当您通过单击它或通过启动|运行从资源管理器等运行它时,当子系统属性为 IMAGE_SUBSYSTEM_WINDOWS_CUI 时,Windows 会自动打开一个控制台窗口。您不需要指定 {$APPTYPE CONSOLE} 指令,这都是关于子系统属性的。
RRUZ 的解决方案是我也在使用的解决方案,但有一个重要区别。我检查父进程的子系统以显示“按 Enter 关闭此窗口”。RUZZ 它的解决方案只在两种情况下有效,当它是 cmd 或 explorer 时。通过简单地检查它的父进程是否具有属性 NOT IMAGE_SUBSYSTEM_WINDOWS_CUI,您可以显示该消息。
但是如何检查exe子系统呢?我在 torry 提示 ( http://www.swissdelphicenter.ch/torry/showcode.php?id=1302 ) 上找到了一个解决方案,以获取 PE Header 信息并将其修改为两个函数:setExeSubSys() 和 getExeSubSys()。使用 setExeSubSys() 我制作了一个小控制台应用程序,这样我就可以在编译后更改 exe 的子系统属性(它只有 50 kb!)。
获得父/潜在进程文件名后,您可以简单地执行以下操作:
//In the very beginning in the app determine the parent process (as fast as is possible).
// later on you can do:
if( getExeSubSys( parentFilename ) <> IMAGE_SUBSYSTEM_WINDOWS_CUI ) then
begin
writeln( 'Press Enter to close the window' );
readln;
end;
这是我制作的两个函数,但它不适用于流(如 torry 示例),我使用我自己的简单单元来处理文件,而没有愚蠢的例外。但基本上我认为你明白了。
设置(以及当你没有指定指向 longint (nil) 的指针时获取):
type
PLongInt = ^LongInt;
function setExeSubSys( fileName : string; pSubSystemId : PLongInt = nil ) : LongInt;
var
signature: DWORD;
dos_header: IMAGE_DOS_HEADER;
pe_header: IMAGE_FILE_HEADER;
opt_header: IMAGE_OPTIONAL_HEADER;
f : TFile;
begin
Result:=-1;
FillChar( f, sizeOf( f ), 0 );
if( fOpenEx( f, fileName, fomReadWrite )) and ( fRead( f, dos_header, SizeOf(dos_header)))
and ( dos_header.e_magic = IMAGE_DOS_SIGNATURE ) then
begin
if( fSeek( f, dos_header._lfanew )) and ( fRead( f, signature, SizeOf(signature))) and ( signature = IMAGE_NT_SIGNATURE ) then
begin
if( fRead( f, pe_header, SizeOf(pe_header))) and ( pe_header.SizeOfOptionalHeader > 0 ) then
begin
if( fRead( f, opt_header, SizeOf(opt_header))) then
begin
if( Assigned( pSubSystemId )) then
begin
opt_header.Subsystem:=pSubSystemId^;
if( fSeek( f, fPos( f )-SizeOf(opt_header) )) then
begin
if( fWrite( f, opt_header, SizeOf(opt_header)) ) then
Result:=opt_header.Subsystem;
end;
end
else Result:=opt_header.Subsystem;
end;
end;
end;
end;
fClose( f );
end;
要得到:
function GetExeSubSystem( fileName : string ) : LongInt;
var
f : TFile;
signature : DWORD;
dos_header: IMAGE_DOS_HEADER;
pe_header : IMAGE_FILE_HEADER;
opt_header: IMAGE_OPTIONAL_HEADER;
begin
Result:=IMAGE_SUBSYSTEM_WINDOWS_CUI; // Result default is console app
FillChar( f, sizeOf( f ), 0 );
if( fOpenEx( f, fileName, fomRead )) and ( fRead( f, dos_header, SizeOf(dos_header)))
and ( dos_header.e_magic = IMAGE_DOS_SIGNATURE ) then
begin
if( fSeek( f, dos_header._lfanew )) and ( fRead( f, signature, SizeOf(signature))) and ( signature = IMAGE_NT_SIGNATURE ) then
begin
if( fRead( f, pe_header, SizeOf(pe_header))) and ( pe_header.SizeOfOptionalHeader > 0 ) then
begin
if( fRead( f, opt_header, SizeOf(opt_header))) then
Result:=opt_header.Subsystem;
end;
end;
end;
fClose( f );
end;
如果您想了解有关子系统的更多信息,只需 google 或访问 MSDN 网站。希望它对任何人都有帮助。
Greetz,欧文·汉杰斯
哇,尼克,这真是令人印象深刻!我已经测试了您的解决方案并且效果很好。
所以你可以做这样的事情:
function isOutputRedirected() : boolean;
var
StdOutHandle : THandle;
bIsNotRedirected : boolean;
FileInfo : TByHandleFileInformation;
begin
StdOutHandle:= GetStdHandle(STD_OUTPUT_HANDLE);
bIsNotRedirected:=( NOT GetFileInformationByHandle(StdOutHandle, FileInfo)
and (GetLastError = ERROR_INVALID_HANDLE));
Result:=( NOT bIsNotRedirected );
end;
function isStartedFromConsole() : boolean;
var
SI: TStartupInfo;
begin
SI.cb := SizeOf(TStartupInfo);
GetStartupInfo(SI);
Result := ((SI.dwFlags and STARTF_USESHOWWINDOW) = 0);
end;
function GetConsoleSize() : _COORD;
var
BufferInfo: TConsoleScreenBufferInfo;
begin
GetConsoleScreenBufferInfo(GetStdHandle(STD_OUTPUT_HANDLE), BufferInfo);
Result.x:=BufferInfo.srWindow.Right - BufferInfo.srWindow.Left + 1;
Result.y:=BufferInfo.srWindow.Bottom - BufferInfo.srWindow.Top + 1;
end;
最后:
var
cKey : Char;
fCursorPos : _COORD;
if( NOT isOutputRedirected() ) and( NOT isStartedFromConsole() ) then
begin
// Windows app starts console.
// Show message in yellow (highlight) and at the bottom of the window
writeln;
fCursorPos:=getConsoleSize();
Dec( fCursorPos.y );
Dec( fCursorPos.x, 40 );
SetConsoleTextAttribute( GetStdHandle(STD_OUTPUT_HANDLE), 14 );
SetConsoleCursorPosition( GetStdHandle(STD_OUTPUT_HANDLE), fCursorPos );
write( '<< Press ENTER to close this window >>' );
read(cKey);
end;
队友的欢呼声!
欧文·汉杰斯
对于程序foo.exe ,制作一个名为foo_runner.bat的批处理文件。不要记录该命令,因为它不是供任何人使用的名称,而是将其用作安装程序制作的任何快捷方式图标的目标。它的内容很简单:
@echo off
%~dp0\foo.exe %*
pause
该%~dp0
部分给出了批处理文件所在的目录,因此您可以确保在批处理文件的目录中运行foo.exe,而不是从搜索路径上的其他位置抓取一个。