本文介绍两种通过Windows进程ID获取窗口句柄的方法:一是使用EnumWindows枚举所有窗口并匹配进程ID;二是利用GetTopWindow和GetNextWindow遍历窗口并验证进程ID。这些方法可以帮助开发者实现对特定进程窗口的操作。
摘要生成于
,由 DeepSeek-R1 满血版支持,
BOOL CALLBACK EnumWindowsProc(HWND hwnd, LPARAM lParam)
EnumWindowsArg *pArg = (EnumWindowsArg *)lParam;
DWORD dwProcessID = 0;
// 通过窗口句柄取得进程ID
::GetWindowThreadProcessId(hwnd, &dwProcessID);
if (dwProcessID == pArg->dwProcessID)
pArg->hwndWindow = hwnd;
// 找到了返回FALSE
return FALSE;
// 没找到,继续找,返回TRUE
return TRUE;
///< 通过进程ID获取窗口句柄
HWND CProcessTimeRestart::GetWindowHwndByPID(DWORD dwProcessID)
HWND hwndRet = NULL;
EnumWindowsArg ewa;
ewa.dwProcessID = dwProcessID;
ewa.hwndWindow = NULL;
EnumWindows(EnumWindowsProc, (LPARAM)&ewa);
if (ewa.hwndWindow)
hwndRet = ewa.hwndWindow;
return hwndRet;
方法二:使用GetTopWindow和GetNextWindow的方式
///< 通过进程ID获取窗口句柄
HWND CProcessTimeRestart::GetWindowHwndByPorcessID(DWORD dwProcessID)
DWORD dwPID = 0;
HWND hwndRet = NULL;
// 取得第一个窗口句柄
HWND hwndWindow = ::GetTopWindow(0);
while (hwndWindow)
dwPID = 0;
// 通过窗口句柄取得进程ID
DWORD dwTheardID = ::GetWindowThreadProcessId(hwndWindow, &dwPID);
if (dwTheardID != 0)
// 判断和参数传入的进程ID是否相等
if (dwPID == dwProcessID)
// 进程ID相等,则记录窗口句柄
hwndRet = hwndWindow;
break;
// 取得下一个窗口句柄
hwndWindow = ::GetNextWindow(hwndWindow, GW_HWNDNEXT);
// 上面取得的窗口,不一定是最上层的窗口,需要通过GetParent获取最顶层窗口
HWND hwndWindowParent = NULL;
// 循环查找父窗口,以便保证返回的句柄是最顶层的窗口句柄
while (hwndRet != NULL)
hwndWindowParent = ::GetParent(hwndRet);
if (hwndWindowParent == NULL)
break;
hwndRet = hwndWindowParent;
// 返回窗口句柄
return hwndRet;
通过以上两种窗口获得的句柄,和使用FindWindow获得的句柄是一样的。
可以通过SendMessage向窗口发送消息
HWND hwnd = ::FindWindow(NULL, "Test");
::SendMessage(hwnd, WM_CLOSE, 0, 0);
2012年-09月-05日
通过进程号的ID获得窗口的句柄
最近几天,工作的需要,打算实现将别人编译好的exe程序,通过自己的程序去调用,然后根据程序中得到的ID号,去得到此进程运行创建的窗口句柄,在网上搜了一段时间,得到了不少启示,再结合自己的实际环境,整合出了以下三种方法,以下将针对每一种方法做出简单的介绍同时给出关键代码,本人所采用的平台为vc6.0,英文企业版。
BOOL CFileOperation::KillProcessFromName(CString strProcessName)
//创建进程快照(TH32CS_SNAPPROCESS表示创建所有进程的快照)
HANDLE hSnapShot = CreateToolhelp32Snapshot(TH32CS_SNAPPROCESS,0);
//PROC
1.窗口类名 窗口句柄 窗口标题 窗口句柄
HWND FindWindow(
LPCTSTR lpClassName, //窗口类名 可用 VC或者VS自带的Spy++查看
LPCTSTR lpWindowName //窗口标题
以 记事本为例, 记事本 窗口类名 为:NotePad, 窗口标题 视按具体情况而定,假设为"新建 文本文档.txt - 记事本"
窗口类名 窗口句柄
TCHAR lpClassName[]=TEXT("NotePad");
HWND hWnd=::FindWindow(lpClassName,NULL);
if(hWnd && IsWindow(hWnd)) ::ShowWindow(hWnd,SW_HIDE);
窗口标题 窗口句柄
TCHAR lpWindowName[]=TEXT("新建 文本文档.txt - 记事本");
HWND hWnd=::FindWindow(NULL,lpWindowName);
if(hWnd && IsWindow(hWnd)) ::ShowWindow(hWnd,SW_HIDE);
DWORD GetWindowThreadProcessId(
HWND hWnd, //目标窗口句柄
LPDWORD lpdwProcessId //返回目标窗口对应进程ID
DWORD dwProcId=0;//存放返回的进程ID
DWORD dwThreadId=0;//存放返回的主线程ID
HWND hWnd=XXXX;//这里省略,可能用任务方式得到一个窗口的句柄.比如用1中的方法.
dwThreadId=GetWindowThreadProcessId(hWnd,&dwProcId);//同时得到进程ID和主线程ID.
3.窗口HAND CWnd
用CWnd::FromHandle(HWND hWnd)函数.很多类都有这个函数.
以记事本为例,进程名为 NOTEPAD.EXE (不一定是大写哦,得到任务管理器是显示而定);
CCheckObject ch;
TCHAR Name[]=TEXT("NOTEPAD.EXE");
DWORD dwProcId=ch.GetProcessId(Name);
5. 进程名 主线程ID
CCheckObject ch;
TCHAR Name[]=TEXT("NOTEPAD.EXE");
DWORD dwThreadId=ch.GetThreadId(Name);
6. 进程名 主窗口句柄
CCheckObject ch;
TCHAR Name[]=TEXT("NOTEPAD.EXE");
HWND hWnd=ch.GetTargetWindowHanle(Name);
7. 其它说明
从CCheckObject类和上面的源码中,不难写出从 进程ID 主线程ID 进程ID 主窗口句柄 主线程ID--->主窗口句柄 等等其它类似转换.
对于主窗口,特点如下:
A. 不能用进程ID,要用线程ID,因为一个进程可能有多个线程,每个线程都可能会有主窗口.
B. 主窗口不会有WS_CHILD属性
C. 主窗口没有父窗口
D. 主窗口一般都有子窗口(这个不是一定的,但是具有普遍性)
2012年-09月-05日
通过进程号的ID获得窗口的句柄
最近几天,工作的需要,打算实现将别人编译好的exe程序,通过自己的程序去调用,然后根据程序中得到的ID号,去得到此进程运行创建的窗口句柄,在网上搜了一段时间,得到了不少启示,再结合自己的实际环境,整合出了以下三种方法,以下将针对每一种方法做出简单的介绍同时给出关键代码,本人所采用的平台为vc6.0,英文企业版。
如题,如何遍历寻找指定进程?如何通过进程ID获取窗口句柄?HWND与CWnd之间是如何转换的?
1、HWND与CWnd之间的转换:使用 GetSafeHwnd 和 CWnd::FromHandle
// 已知 CWnd 获取 HWND
CWnd* pCWnd;
HWND hwnd = pCWnd->m_hWnd;
HWND hwnd1 = pCWnd->GetSafeHwnd();
// 已知 HWND 获取 CWnd
HWND hwnd2;
CWnd* pCwnd1 = CW
通过进程ID获取窗口句柄可以使用Windows API中的EnumWindows函数结合GetWindowThreadProcessId函数来实现。
EnumWindows函数是用于枚举所有顶级窗口的函数,它会将每个窗口的句柄传入一个回调函数中进行处理。而GetWindowThreadProcessId函数用于获取给定窗口的进程ID。
首先,我们可以定义一个回调函数来处理EnumWindows函数传入的窗口句柄。在回调函数中,我们需要利用GetWindowThreadProcessId函数来获取每个窗口的进程ID,并将其与我们想要获取的目标进程ID进行比较。当匹配到目标进程ID时,我们可以将该窗口句柄保存下来。
以下是一个简单的示例代码:
```c++
#include <iostream>
#include <windows.h>
DWORD targetProcessId;
HWND targetWindowHandle = nullptr;
BOOL CALLBACK EnumWindowsProc(HWND hwnd, LPARAM lParam)
DWORD currentProcessId;
GetWindowThreadProcessId(hwnd, ¤tProcessId);
if (currentProcessId == targetProcessId)
targetWindowHandle = hwnd;
return FALSE; // 返回FALSE以停止枚举
return TRUE; // 返回TRUE以继续枚举
HWND GetWindowHandleByProcessId(DWORD processId)
targetProcessId = processId;
EnumWindows(EnumWindowsProc, 0);
return targetWindowHandle;
int main()
DWORD processId = 1234; // 目标进程ID
HWND windowHandle = GetWindowHandleByProcessId(processId);
if (windowHandle != nullptr)
std::cout << "窗口句柄: " << windowHandle << std::endl;
std::cout << "未找到匹配的窗口句柄" << std::endl;
return 0;
在示例代码中,我们定义了一个名为`GetWindowHandleByProcessId`的函数来实现通过进程ID获取窗口句柄的功能。该函数接受一个参数`processId`,即目标进程的ID。在函数内部,我们首先将目标进程ID保存起来,然后调用`EnumWindows`函数来枚举所有顶级窗口。在回调函数`EnumWindowsProc`中,我们使用`GetWindowThreadProcessId`函数获取当前窗口的进程ID并与目标进程ID进行比较。如果匹配成功,则将当前窗口句柄保存下来。最后,在`main`函数中,我们可以调用`GetWindowHandleByProcessId`函数来获取目标进程的窗口句柄,并输出结果。
需要注意的是,以上示例代码仅演示了通过进程ID获取单个窗口句柄的方法。如果目标进程存在多个窗口,您可能需要进一步定义您的需求来确定具体需要获取哪个窗口的句柄。