C/C++ 遍历窗口标题类名
遍历每个进程,一次查找进程下的窗口,找到窗口标题为 “” ,窗口类名为 “RunDll” 的窗口。如果找到返回 true ,没找到返回 false。
#pragma region 依赖
typedef struct EnumHWndsArg{
std::vector<HWND> *vecHWnds;
DWORD dwProcessId;
}EnumHWndsArg, *LPEnumHWndsArg;
// 判断窗口是否属于目标进程
BOOL CALLBACK lpEnumFunc(HWND hwnd, LPARAM lParam){
EnumHWndsArg *pArg = (LPEnumHWndsArg)lParam;
DWORD processId;
// 检索窗口线程标识符
GetWindowThreadProcessId(
hwnd, // 窗口句柄
&processId // 接收 PID 的指针
);
// 如果这个 HWND 属于这个 PID ,则加入到 vecHWnds 数组末尾
if (processId == pArg->dwProcessId){pArg->vecHWnds->push_back(hwnd);}
return TRUE;
}
// 根据 PID 获取 HWND
void GetHWndsByProcessID(DWORD processID, std::vector<HWND> &vecHWnds){
EnumHWndsArg wi;
wi.dwProcessId = processID;
wi.vecHWnds = &vecHWnds;
// 枚举所有顶级窗口
EnumWindows(
lpEnumFunc, // 回调函数指针
(LPARAM)&wi // 传递给回调函数的值
);
}
#pragma endregion
bool CSOL_dh()
{
HANDLE hProcessSnap = CreateToolhelp32Snapshot(TH32CS_SNAPPROCESS,0);
PROCESSENTRY32 process = {sizeof(PROCESSENTRY32)};
// 遍历进程
while (Process32Next(hProcessSnap,&process)){
string processName = process.szExeFile; // 进程名
std::vector<HWND> vecHWnds; // 进程下的窗体句柄数组
GetHWndsByProcessID(process.th32ProcessID,vecHWnds);
// 获取 HWND 窗口标题、窗口类名
TCHAR szBuf_title[MAX_PATH];
TCHAR szBuf_class[MAX_PATH];
for (const HWND &h : vecHWnds){
GetWindowText(
h, // 窗口句柄
szBuf_title, // 接收窗口标题的缓冲区指针
MAX_PATH // 缓冲区字节大小
);
GetClassName(
h, // 窗口句柄
szBuf_class, // 接收窗口类名的缓冲区指针
MAX_PATH // 缓冲区字节大小
);
// 判断是否有符合要求的窗体
if(strcmp(szBuf_title,"") == 0 && strcmp(szBuf_class,"RunDLL") == 0){
return true;
}
// 输出结果
//cout << "szBuf_title = " << szBuf_title << endl;
//cout << "szBuf_class = " << szBuf_class << endl;
//cout << "--------------------------------------------" << endl;
}
}
return false;
}
API——获取鼠标位置的窗口句柄
#include <Windows.h>
#define MAX_TEXT_LEN 255
BOOL CALLBACK EnumChildProcess(HWND hwnd, LPARAM lParam)
{
if (hwnd == NULL) {
return FALSE;
}
BOOL ret;
RECT rect;
ret = GetWindowRect(hwnd, &rect);
if (!ret) {
printf("GetWindowRect hwnd=%p -> fail(%ld)\n", hwnd, GetLastError());
}
else {
//printf("GetWindowRect hwnd = %p -> rect=(left=%ld, top=%ld, right=%ld, bottom=%ld)\n", hwnd, rect.left, rect.top, rect.right, rect.bottom);
ret = PtInRect(&rect, *(POINT *)lParam);
if (ret) {
printf("GetWindowRect hwnd = %p -> rect=(left=%ld, top=%ld, right=%ld, bottom=%ld)\n", hwnd, rect.left, rect.top, rect.right, rect.bottom);
//printf("PtInRect\n");
/*
WINUSERAPI int WINAPI GetWindowText(
_In_ HWND hWnd,
_Out_writes_(nMaxCount) LPTSTR lpString, //可能是标题名或者file:///打头的文件完整路径
_In_ int nMaxCount
);
如果函数成功,返回值是拷贝的字符串的字符个数,不包括中断的空字符;如果窗口无标题栏或文本,或标题栏为空,或窗口或控制的句柄无效,则返回值为零。若想获得更多错误信息,请调用GetLastError函数。
*/
TCHAR windowText[MAX_TEXT_LEN];
int lenRet = GetWindowText(hwnd, windowText, MAX_TEXT_LEN);
if (lenRet == 0 && GetLastError() != 0) {
//GetLastError()〖0〗-操作成功完成
printf("GetWindowText hwnd=%p -> fail(%ld)\n", hwnd, GetLastError());
}
else {
_tprintf(_T("GetWindowText hwnd=%p -> windowText=%s, lenRet=%d\n"), hwnd, windowText, lenRet);
}
/*
WINUSERAPI int WINAPI GetClassNameW(
_In_ HWND hWnd,
_Out_writes_to_(nMaxCount, return) LPWSTR lpClassName,
_In_ int nMaxCount
);
如果函数成功,返回值为拷贝到指定缓冲区的字符个数:如果函数失败,返回值为0。若想获得更多错误信息,请调用GetLastError函数。
*/
TCHAR className[MAX_TEXT_LEN];
lenRet = GetClassName(hwnd, className, MAX_TEXT_LEN);
if (lenRet == 0) {
printf("GetClassName hwnd=%p -> fail(%ld)\n", hwnd, GetLastError());
}
else {
_tprintf(_T("GetClassName hwnd=%p -> className=%s, lenRet=%d\n"), hwnd, className, lenRet);
}
/*
找出某个窗口的创建者(线程或进程),返回创建者的标志符
哪个线程创建了这个窗口,返回的就是这个线程的id号 (进程只有一个线程的话,那么线程标志符与进程标志符就是指同一个标志符)
WINUSERAPI DWORD WINAPI GetWindowThreadProcessId(
_In_ HWND hWnd,
_Out_opt_ LPDWORD lpdwProcessId //进程号的存放地址(变量地址)
);
返回线程号
*/
DWORD dwProcessId;
DWORD dwThreadId = GetWindowThreadProcessId(hwnd, &dwProcessId);
printf("GetWindowThreadProcessId hwnd=%p -> processId=%ld, threadId=%ld\n", hwnd, dwProcessId, dwThreadId);
/*
WINUSERAPI UINT WINAPI GetWindowModuleFileName(
_In_ HWND hwnd,
_Out_writes_to_(cchFileNameMax, return) LPTSTR pszFileName, //模块完整路径
_In_ UINT cchFileNameMax
);
返回值是复制到缓冲区的字符总数。
*/
TCHAR fileName[MAX_PATH];
lenRet = GetWindowModuleFileName(hwnd, fileName, MAX_PATH);
if (lenRet == 0) {
//错误码〖126〗-找不到指定的模块。
printf("GetWindowModuleFileName hwnd=%p -> fail(%ld)\n", hwnd, GetLastError());
} else {
_tprintf(_T("GetWindowModuleFileName hwnd=%p -> fileName=%s\n"), hwnd, fileName);
}
/*
WINUSERAPI BOOL WINAPI GetWindowInfo(
_In_ HWND hwnd,
_Inout_ PWINDOWINFO pwi
);
typedef struct tagWINDOWINFO
{
DWORD cbSize;
RECT rcWindow;
RECT rcClient;
DWORD dwStyle;
DWORD dwExStyle;
DWORD dwWindowStatus;
UINT cxWindowBorders;
UINT cyWindowBorders;
ATOM atomWindowType;
WORD wCreatorVersion;
} WINDOWINFO, *PWINDOWINFO, *LPWINDOWINFO;
*/
WINDOWINFO windowInfo;
windowInfo.cbSize = sizeof(WINDOWINFO);
ret = GetWindowInfo(hwnd, &windowInfo);
if (!ret) {
printf("GetWindowInfo hwnd=%p -> fail(%ld)\n", hwnd, GetLastError());
}
else {
printf("GetWindowInfo hwnd=%p -> dwStyle=%ld, dwExStyle=%ld, dwWindowStatus=%ld, cxWindowBorders=%d, cyWindowBorders=%d, wCreatorVersion=%d\n", hwnd, windowInfo.dwStyle, windowInfo.dwExStyle, windowInfo.dwWindowStatus, windowInfo.cxWindowBorders, windowInfo.cyWindowBorders, windowInfo.wCreatorVersion);
}
printf("\n");
}
}
return TRUE;
}
int main()
{
BOOL ret;
while (true) {
/*
typedef struct tagPOINT
{
LONG x;
LONG y;
} POINT, *PPOINT, NEAR *NPPOINT, FAR *LPPOINT;
*/
POINT point;
ret = GetCursorPos(&point);
if (!ret) {
printf("GetCursorPos -> fail(%ld)\n", GetLastError());
}
else {
printf("GetCursorPos -> (%ld, %ld)\n", point.x, point.y);
//获取桌面句柄
HWND desktopHwnd = GetDesktopWindow();
/*
BOOL EnumChildWindows(
HWND hWndParent, // handle to parent window // 父窗口句柄
WNDENUMPROC lpEnumFunc, // callback function // 回调函数的地址
LPARAM lParam // application-defined value // 你自已定义的参数
);
直到调用到最个一个子窗口被枚举或回调函数返回一个false,否则将一直自动枚举下去。
*/
ret = EnumChildWindows(desktopHwnd, EnumChildProcess, (LPARAM)&point);
}
/*
WINBASEAPI VOID WINAPI Sleep(
_In_ DWORD dwMilliseconds
);
Sleep会将线程挂起,把CPU让给其它线程,单位是毫秒
*/
Sleep(20000);
}
system("pause");
return 0;
}
文章出处:https://www.cnblogs.com/LyShark/articles/15020012.html
本博客所有文章除特别声明外,均采用 BY-NC-SA 许可协议。转载请注明出处!
本博客所有文章除特别声明外,均采用 BY-NC-SA 许可协议。转载请注明出处!
标签:
Visual C++
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
· 基于Microsoft.Extensions.AI核心库实现RAG应用
· Linux系列:如何用heaptrack跟踪.NET程序的非托管内存泄露
· TypeScript + Deepseek 打造卜卦网站:技术与玄学的结合
· 阿里巴巴 QwQ-32B真的超越了 DeepSeek R-1吗?
· 【译】Visual Studio 中新的强大生产力特性
· 10年+ .NET Coder 心语 ── 封装的思维:从隐藏、稳定开始理解其本质意义
· 【设计模式】告别冗长if-else语句:使用策略模式优化代码结构