如何获取外部EXE窗口的handle

这是我在一个论坛的讨论稿,我想这些情况在实践程序设计过程中会常碰到,所以就将这些讨论资料写了下来:

作者:e梦缘

请问用什么方法可以运行一个外部的EXE并返回它的Handle?
WinExec 返回不是该EXE的Handle
ShellExecute, ShellExecuteEx 也不是
CreateProcess 返回的是 Process Handle
到底要如何才可以返回和 GetWindow 所返回的一样的 Handle?

Re:

可以用FindWindow()函数,它能够解决这个问题,它有两个参数:lpClassName:程序的类名;lpWindowName:程序窗体的标题。

例如:

procedure TForm1.Button1Click(Sender: TObject);
begin
ShellExecute(handle,'open','notepad.exe',nil,nil,SW_ShowNormal);
end;

procedure TForm1.Button2Click(Sender: TObject);
var
ExeHandle:Thandle;
Begin
//获得句柄
ExeHandle:=FindWindow('notepad',nil);//'');//返回句柄
//关闭程序
if ExeHandle<>0 then
SendMessage(ExeHandle,WM_Close,0,0)
Else
Application.MessageBox('没有打开"记事本"程序!','提示',
MB_IconInformation+MB_OK);
end;

?
FindWindow() 方法不是我想要的,因为这里的lpWindowName 具有不确定性,没有办法准确地找到。
我希望能在执行这支外部程序里就可以得到它的 Handle.
比如有没有办法,通过Process Handle得到它有
Application Handle 以及 MainForm Handle

Re:

您可以通过lpClassName:程序的类名来准确地找到。

ExeHandle:=FindWindow('notepad',nil);//返回句柄

移至頂端function ExecuteFile(const FileName, Params, DefaultDir: String;

ShowCmd: Integer): THandle;

var

zFileName, zParams, zDir: array[0..79] of Char;

begin

Result := ShellExecute(Application.MainForm.Handle, nil,

StrPCopy(zFileName, FileName), StrPCopy(zParams, Params),

StrPCopy(zDir, DefaultDir), ShowCmd);

end;

procedure TForm1.Button3Click(Sender: TObject);
var
hwd:thandle;
begin

hwd:=ExecuteFile('notepad.exe','','',SW_ShowNormal);
if hwd<>0 then showmessage('哈哈~~'+#13+'我取到handle了,是>>>'+inttostr(hwd));

end;

 

?

谢谢您的热心帮助,
但ShellExecute得到的并不是Handle
用FindWindow得到的才是真正的Handle
你可以试一下.
最简单的方法就是向你得到的Handle发一个WM_QUIT消息,看它是否会被关闭.
如:
SendMessage(AHandle, WM_QUIT, 0, 0);

 

Re:

我看了很多delphi的资料,好像只有FindWindow()函数是 可以实现的,至于是否还有其它的方法,我 不太清楚!

 

最后结果:

我已从大富翁论坛上得到了满意的结论了,在这和大家分享一下:
(大富翁论坛: http://www.delphibbs.com/)

type
PEnumInfo = ^TEnumInfo;
TEnumInfo = record
ProcessID : DWORD;
HWND : THandle;
end;

function EnumWindowsProc(Wnd: DWORD; var EI: TEnumInfo): Bool; stdcall;
var
PID : DWORD;
begin
GetWindowThreadProcessID(Wnd, @PID);
Result := (PID <> EI.ProcessID) or
(not IsWindowVisible(WND)) or
(not IsWindowEnabled(WND));

if not result then EI.HWND := WND;
end;

function FindMainWindow(PID: DWORD): DWORD;
var
EI : TEnumInfo;
begin
EI.ProcessID := PID;
EI.HWND := 0;
EnumWindows(@EnumWindowsProc, Integer(@EI));
Result := EI.HWND;
end;


procedure TForm1.Button1Click(Sender: TObject);
var
SI : TStartupInfo;
PI : TProcessInformation;
H : THandle;
S : String;
begin
ZeroMemory(@SI, SizeOf(SI));
ZeroMemory(@PI, SizeOf(PI));
SI.cb := SizeOf(SI);
if CreateProcess(nil,'CALC.EXE', nil, nil, FALSE, 0 ,nil,nil, SI, PI) then
begin
//注意!
WaitForInputIdle(PI.hProcess, INFINITE);

H := FindMainWindow(PI.dwProcessID);
if H > 0 then
begin
SetLength(S, 255);
GetWindowText(H, PChar(S), 255);
SetLength(S, StrLen(PChar(S)));
ShowMessage(S);
end;

CloseHandle(PI.hProcess);
CloseHandle(PI.hThread);
end;
end;
end.

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
freepdb.cmd文件解读 说明:@字符后面的是注释 if "%1"=="" ( @如果"%1"表示输入的参数如果为空,则进行如下处理 echo Usage: freepdb filename echo This will free all handles of VS2003 on filename.pdb echo Requires the Handle utility from Windows Sysinternals goto :eof ) cd /d "%~dp0" @进入freepdb.cmd文件所在的目录,因为需要使用的handle.exe和 @freepdb.cmd在同一个目录下 @以下部分说明:阅读时,请先查询dos命令 for的使用,第一个for循环来获取指定pdb文件的信息。 for /f "tokens=2,3 skip=5 delims=:" %%a in ('handle.exe -p devenv.exe "%1.pdb"') do ( for /f "tokens=1,4" %%c in ("%%a %%b") do ( handle -p %%c -c %%d -y >a.txt ) ) exit /b 0 @第一个for循环的含义是:运行in后面的括号里的命令语句,将结果按照/f 后面的命令来筛选出来, @@将到的结果分别放入了%%a, %%b中 @运行in后面括号中的语句后,得到的结果如下(调试我自己的程序打印出来的信息): @devenv.exe pid: 8960 type: File B70:E:\work\MainClient\new_src\Release\3DStock.pdb @然后取第2个字符串和第3个字符串,分别是(因为是通过“:”号来分割字符串的): @8960 type @File B70 @执行第2个循环语句 @第2个for循环,是从%%c(由%%a, %%b所合成的)中,取第一个字符串(默认通过空格来分割字符串)和第 @4个字符串 @运行in后面括号中的语句结果如下: @ 8960 type File B70 @然后取第1个字符串和第4个字符串,分别放入%%c, %%d中(8960是进程的id,B70是文件的句柄) @8960 @B70 @然后执行handle -p %%c -c %%d -y >a.txt命令 @即执行:handle -p 8960 -c B70 -y, 并将处理的结果存入a.txt ,存入文件是为了看到处理结果 @此命令是,杀掉指定进程ID的使用的文件句柄

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值