通过进程名称获取进程ID、通过子进程ID获取其父进程ID及通过进程ID获取进程名称

最近一个MFC项目里有个需求是:需要根据一个进程名称获取其父进程名称。于是查了些资料整理出来份代码,并进行了些测试结果OK。直接拿来即可当工具类使用,以下贴上代码:

头文件GetProcessInfo.h:

#pragma once
#include <wtypes.h>

#define ProcessBasicInformation 0

typedef struct
{
	DWORD ExitStatus;
	DWORD PebBaseAddress;
	DWORD AffinityMask;
	DWORD BasePriority;
	ULONG UniqueProcessId;
	ULONG InheritedFromUniqueProcessId;
}   PROCESS_BASIC_INFORMATION;

typedef LONG (WINAPI *PROCNTQSIP)(HANDLE,UINT,PVOID,ULONG,PULONG);

class GetProcessInfo
{
public:
	GetProcessInfo(void);
	~GetProcessInfo(void);

	//通过进程名称获取进程ID,同时也可通过传个引用参数获取到父进程ID,同下面GetParentProcessId获取的结果一致
	DWORD GetProcessIdFromName(CString strProcessName, DWORD &dwParentProcessId);

	//通过子进程ID获取其父进程ID
	DWORD GetParentProcessId(DWORD dwChildProcessId);

	//通过进程ID获取进程名称
	CString GetProcessNameFromId(DWORD dwProcessId);
};

源文件GetProcessInfo.cpp内容:

#include "stdafx.h"
#include "GetProcessInfo.h"
#include <tlhelp32.h>

#pragma comment (lib,"Psapi.lib")

#include <Psapi.h>

GetProcessInfo :: GetProcessInfo(void)
{
}

GetProcessInfo :: ~GetProcessInfo(void)
{
}

DWORD GetProcessInfo :: GetProcessIdFromName(CString strProcessName, DWORD &dwParentProcessId)
{
	DWORD dwProcessID =0;

	//进行一个进程快照
	HANDLE hProcessSnap = CreateToolhelp32Snapshot(TH32CS_SNAPPROCESS,0);
	if (hProcessSnap == INVALID_HANDLE_VALUE)
	{
		OutputDebugString(_T("进程快照失败!"));
		return FALSE;
	}

	PROCESSENTRY32 pe;
	pe.dwSize = sizeof(pe);
	BOOL bProcess = Process32First(hProcessSnap,&pe);
	while (bProcess)
	{
		if (strProcessName.CompareNoCase(pe.szExeFile) == 0)
		{
			dwProcessID = pe.th32ProcessID;
			dwParentProcessId = pe.th32ParentProcessID; //pe结构中包含有父进程的ID
		}
		bProcess = Process32Next(hProcessSnap,&pe);
	}

	CloseHandle(hProcessSnap);

	return dwProcessID;
}

DWORD GetProcessInfo :: GetParentProcessId(DWORD dwChildProcessId)
{
	//NtQueryInformationProcess函数的使用需要加载进ntdll.dll
	PROCNTQSIP NtQueryInformationProcess = (PROCNTQSIP)GetProcAddress(GetModuleHandle(_T("ntdll.dll")), "NtQueryInformationProcess");

	if(!NtQueryInformationProcess)
	{
		OutputDebugString(_T("ntdll.dll中检索NtQueryInformationProcess失败!"));
	}

	DWORD dwParentProcessId = 0;
	LONG status;
	PROCESS_BASIC_INFORMATION pbi;

	HANDLE hProcess = OpenProcess(PROCESS_QUERY_INFORMATION,FALSE,dwChildProcessId);
	if (!hProcess)
	{
		OutputDebugString(_T("OpenProcess Error!"));
		return FALSE;
	}

	status = NtQueryInformationProcess( hProcess, ProcessBasicInformation, (PVOID)&pbi, sizeof(PROCESS_BASIC_INFORMATION), NULL);

	if  (!status)
	{
		dwParentProcessId = (DWORD)pbi.InheritedFromUniqueProcessId;
		CString strParentID;
		strParentID.Format(_T("%d"), dwParentProcessId);
		OutputDebugString(_T("ParentProcessID:")+strParentID);	
	}

	return dwParentProcessId;
}

CString GetProcessInfo :: GetProcessNameFromId(DWORD dwProcessId)
{
	CString strProcessName;
	HANDLE hProcess = OpenProcess( PROCESS_QUERY_INFORMATION|PROCESS_VM_READ, FALSE, dwProcessId);
	if (NULL != hProcess )
	{
		
		HMODULE hMod;
		DWORD cbNeeded;
		if ( EnumProcessModules( hProcess, &hMod, sizeof(hMod), &cbNeeded) )
		{
			GetModuleBaseName( hProcess, hMod, strProcessName.GetBuffer(MAX_PATH), MAX_PATH);
			strProcessName.ReleaseBuffer();
		}
	}
	return strProcessName;
}

用到时,直接将该头文件和源文件引入项目,在需要用到的地方定义一个GetProcessInfo对象,然后传参数调用其相关函数即可。


  • 1
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
VC获得进程ID获得主线程ID获得窗口句柄获得主窗口获得进程名 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); 2.窗口句柄 进程ID 窗口句柄 主线程ID 要使用到的函数: 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)函数.很多类都有这个函数. 4.进程进程ID (注:进程名,即在"任务管理器"中看到的名字) 用CCheckObject类(详细实现源文件); 例子: 以记事本为例,进程名为 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. 主窗口一般都有子窗口(这个不是一定的,但是具有普遍性)

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值