枚举进程的方法有许多,简单说一下它们的优缺点。
1.利用CreateToolhelp32Snapshot来枚举,这是最常用,也最简单的方法了,不用多说。
2.利用ZwQuerySystemInformation从ntdll.dll中获取,需要定义结构体SYSTEM_PROCESS_INFORMATION,还有对应的函数指针,但是要注意一点,申请内存之后,要判断ZwQuerySystemInformation的返回值是不是STATUS_INFO_LENGTH_MISMATCH,如果是,说明内存不足,要重新申请更大的。
3.利用PSAPI.DLL中提供的EnumProcesses,EnumProcessModules等函数来获取,这种方法是去打开进程,因此对于权限的要求特别高,即使加上提权代码,或者以管理员权限运行,仍然有许多进程信息得不到,但是它可以得到部分进程完整路径。根据自己的需求来选择。
4.利用服务中的WTSOpenServerA,WTSEnumerateProcessesA函数来得到,下面重点说说这种方法。
#include <windows.h> #include <stdio.h> #include <Wtsapi32.h> #pragma comment (lib,"Wtsapi32.lib") int main() { WCHAR* szServerName = L""; //win10 不需要 HANDLE WtsServerHandle = WTSOpenServer(szServerName); // 然后开始遍历终端服务器上的所有进程,这里我们是指本机的所有进程. PWTS_PROCESS_INFO pWtspi; DWORD dwCount; if (!WTSEnumerateProcesses(WtsServerHandle, 0, 1, &pWtspi, &dwCount)) { int a = GetLastError(); return 0; } for (DWORD i = 0; i < dwCount; i++) { printf("ProcessID: %d (%ls) ", pWtspi[i].ProcessId, pWtspi[i].pProcessName); } getchar(); }
win10下已经提供了Wtsapi32.lib,Wtsapi32.dll等,因此程序简化了很多,只是对于服务名szServerName,看资料以前的操作系统要主机的名字,NetBios指定的终端服务名,可以通过在控制台命令行下用 nbtstat –n 来获取本机NetBios名,我这样做了之后,却发现无法枚举,调用GetLastError()函数,错误是0x000006BA ,RPC 服务器不可用。将RPC重启后还是不行,最后上网查了,原来win10下不需要这个名字。终于枚举成功了。
对于比较旧的操作系统,建议用函数指针的方法实现:
typedef HANDLE(WINAPI *WTSOPENSERVER)(LPTSTR ServerName); typedef BOOL(WINAPI *WTSENUMERATEPROCESSES) ( HANDLE ServerHandle, // WTSOpenServer返回的句柄. DWORD Reserved, // 保留值 0. DWORD Version, // 指定枚举要求的版本,必须为 1. PWTS_PROCESS_INFO* ProcessInfo, // 存放我们要的进程名和进程id. DWORD* Count // 存放ppProcessInfo里WTS_PROCESS_INFO结构数量指针. ); HMODULE WtsApi32Handle = LoadLibrary(L"wtsapi32.dll"); WTSOPENSERVER WtsOpenServer = (WTSOPENSERVER) GetProcAddress(WtsApi32Handle, "WTSOpenServerA"); WTSENUMERATEPROCESSES WtsEnumerateProcesses = (WTSENUMERATEPROCESSES) GetProcAddress(WtsApi32Handle, "WTSEnumerateProcessesA");