当前位置: 首页 > news >正文

Windows系统编程项目(一)进程管理器

本项目将通过MFC实现一个进程管理器,如下图详细信息页所示:

一.首先创建一个基于对话框的MFC项目,在静态库中使用MFC

二.在项目默认的对话框中添加一个列表

三.列表添加变量

四.初始化列表

1.设置列表风格和表头

2.填充列表内容

我们需要在列表中填充操作系统中运行的各种进程的信息,因此我们需要一个函数帮助我们遍历运行的进程并获取相关信息

注意:遍历进程时所使用的API,我们需要在对话框头文件中包含相应的头文件

首先声明一个遍历进程填充函数

然后实现这个遍历进程填充函数

3.完善列表初始化

完成以上代码书写以后,便实现了一个简单的没有功能的任务管理器

五.功能实现

接下来实现任务管理器的右键可以结束进程或打开线程列表。线程列表右键结束线程或挂起恢复线程等等功能

1.列表添加一个右键消息处理函数

2.添加一个菜单

3.列表实现点击右键弹出菜单

4.实现刷新功能

5.实现结束进程功能

6.实现创建进程功能

以下是我手搓的代码,建议诸位参考上文的演示,自己也写一个,有助于加深理解


// 任务管理器Dlg.cpp: 实现文件
//

#include "pch.h"
#include "framework.h"
#include "任务管理器.h"
#include "任务管理器Dlg.h"
#include "afxdialogex.h"

#ifdef _DEBUG
#define new DEBUG_NEW
#endif


// 用于应用程序“关于”菜单项的 CAboutDlg 对话框

class CAboutDlg : public CDialogEx
{
public:
	CAboutDlg();

// 对话框数据
#ifdef AFX_DESIGN_TIME
	enum { IDD = IDD_ABOUTBOX };
#endif

	protected:
	virtual void DoDataExchange(CDataExchange* pDX);    // DDX/DDV 支持

// 实现
protected:
	DECLARE_MESSAGE_MAP()
};

CAboutDlg::CAboutDlg() : CDialogEx(IDD_ABOUTBOX)
{
}

void CAboutDlg::DoDataExchange(CDataExchange* pDX)
{
	CDialogEx::DoDataExchange(pDX);
}

BEGIN_MESSAGE_MAP(CAboutDlg, CDialogEx)
END_MESSAGE_MAP()


// C任务管理器Dlg 对话框



C任务管理器Dlg::C任务管理器Dlg(CWnd* pParent /*=nullptr*/)
	: CDialogEx(IDD_MY_DIALOG, pParent)
{
	m_hIcon = AfxGetApp()->LoadIcon(IDR_MAINFRAME);
}

void C任务管理器Dlg::DoDataExchange(CDataExchange* pDX)
{
	CDialogEx::DoDataExchange(pDX);
	DDX_Control(pDX, IDC_LIST1, m_ProcessList);
}

BEGIN_MESSAGE_MAP(C任务管理器Dlg, CDialogEx)
	ON_WM_SYSCOMMAND()
	ON_WM_PAINT()
	ON_WM_QUERYDRAGICON()
	ON_NOTIFY(NM_RCLICK, IDC_LIST1, &C任务管理器Dlg::OnRclickList1)
	ON_COMMAND(ID_32771, &C任务管理器Dlg::OnProcessList)
	ON_COMMAND(ID_32772, &C任务管理器Dlg::OnKillProcess)
	ON_COMMAND(ID_32773, &C任务管理器Dlg::OnCreateProcess)
END_MESSAGE_MAP()


// C任务管理器Dlg 消息处理程序

BOOL C任务管理器Dlg::OnInitDialog()
{
	CDialogEx::OnInitDialog();

	// 将“关于...”菜单项添加到系统菜单中。

	// IDM_ABOUTBOX 必须在系统命令范围内。
	ASSERT((IDM_ABOUTBOX & 0xFFF0) == IDM_ABOUTBOX);
	ASSERT(IDM_ABOUTBOX < 0xF000);

	CMenu* pSysMenu = GetSystemMenu(FALSE);
	if (pSysMenu != nullptr)
	{
		BOOL bNameValid;
		CString strAboutMenu;
		bNameValid = strAboutMenu.LoadString(IDS_ABOUTBOX);
		ASSERT(bNameValid);
		if (!strAboutMenu.IsEmpty())
		{
			pSysMenu->AppendMenu(MF_SEPARATOR);
			pSysMenu->AppendMenu(MF_STRING, IDM_ABOUTBOX, strAboutMenu);
		}
	}

	// 设置此对话框的图标。  当应用程序主窗口不是对话框时,框架将自动
	//  执行此操作
	SetIcon(m_hIcon, TRUE);			// 设置大图标
	SetIcon(m_hIcon, FALSE);		// 设置小图标

	// TODO: 在此添加额外的初始化代码
	m_ProcessList.SetExtendedStyle(LVS_EX_AUTOSIZECOLUMNS | LVS_EX_COLUMNSNAPPOINTS | LVS_EX_FLATSB | LVS_EX_GRIDLINES | LVS_EX_FULLROWSELECT);
	m_ProcessList.InsertColumn(0, L"进程名", LVCFMT_LEFT, 150 );
	m_ProcessList.InsertColumn(1, L"进程ID", LVCFMT_LEFT, 100);
	m_ProcessList.InsertColumn(2, L"父进程ID", LVCFMT_LEFT, 100);
	m_ProcessList.InsertColumn(3, L"线程数", LVCFMT_LEFT, 100);
	m_ProcessList.InsertColumn(4, L"优先级", LVCFMT_LEFT, 100);
	m_ProcessList.InsertColumn(5, L"进程路径", LVCFMT_LEFT, 300);
	InitProcessList();
	return TRUE;  // 除非将焦点设置到控件,否则返回 TRUE
}

void C任务管理器Dlg::OnSysCommand(UINT nID, LPARAM lParam)
{
	if ((nID & 0xFFF0) == IDM_ABOUTBOX)
	{
		CAboutDlg dlgAbout;
		dlgAbout.DoModal();
	}
	else
	{
		CDialogEx::OnSysCommand(nID, lParam);
	}
}

// 如果向对话框添加最小化按钮,则需要下面的代码
//  来绘制该图标。  对于使用文档/视图模型的 MFC 应用程序,
//  这将由框架自动完成。

void C任务管理器Dlg::OnPaint()
{
	if (IsIconic())
	{
		CPaintDC dc(this); // 用于绘制的设备上下文

		SendMessage(WM_ICONERASEBKGND, reinterpret_cast<WPARAM>(dc.GetSafeHdc()), 0);

		// 使图标在工作区矩形中居中
		int cxIcon = GetSystemMetrics(SM_CXICON);
		int cyIcon = GetSystemMetrics(SM_CYICON);
		CRect rect;
		GetClientRect(&rect);
		int x = (rect.Width() - cxIcon + 1) / 2;
		int y = (rect.Height() - cyIcon + 1) / 2;

		// 绘制图标
		dc.DrawIcon(x, y, m_hIcon);
	}
	else
	{
		CDialogEx::OnPaint();
	}
}

//当用户拖动最小化窗口时系统调用此函数取得光标
//显示。
HCURSOR C任务管理器Dlg::OnQueryDragIcon()
{
	return static_cast<HCURSOR>(m_hIcon);
}

void C任务管理器Dlg::InitProcessList()
{
	HANDLE hSnp = CreateToolhelp32Snapshot(TH32CS_SNAPPROCESS, 0);
	PROCESSENTRY32 pe = { sizeof(PROCESSENTRY32) };
	WCHAR wcth32ProcessID[MAX_PATH];//进程ID
	WCHAR wcth32ParentProcessID[MAX_PATH];//父进程ID 
	WCHAR wccntThreads[MAX_PATH];//线程数
	WCHAR wcpcPriClassBase[MAX_PATH];//优先级
	WCHAR wcProcessPath[MAX_PATH];//进程路径
	BOOL Success = Process32First(hSnp, &pe);
	DWORD Index = 0;
	while (Success)
	{
		HANDLE Process = OpenProcess(PROCESS_ALL_ACCESS, FALSE, pe.th32ProcessID);
		GetModuleFileNameEx(Process, NULL, wcProcessPath, MAX_PATH);
		if (Process == NULL)
		{
			swprintf(wcProcessPath, L"System Module");
		}
		swprintf(wcth32ProcessID, L"%d", pe.th32ProcessID);
		swprintf(wcth32ParentProcessID, L"%d", pe.th32ParentProcessID);
		swprintf(wccntThreads, L"%d", pe.cntThreads);
		swprintf(wcpcPriClassBase, L"%d", pe.pcPriClassBase);
		m_ProcessList.InsertItem(Index, pe.szExeFile);
		m_ProcessList.SetItemText(Index, 1, wcth32ProcessID);
		m_ProcessList.SetItemText(Index, 2, wcth32ParentProcessID);
		m_ProcessList.SetItemText(Index, 3, wccntThreads);
		m_ProcessList.SetItemText(Index, 4, wcpcPriClassBase);
		m_ProcessList.SetItemText(Index, 5, wcProcessPath);
		Index++;
		Success = Process32Next(hSnp, &pe);
	}



}



void C任务管理器Dlg::OnRclickList1(NMHDR* pNMHDR, LRESULT* pResult)
{
	LPNMITEMACTIVATE pNMItemActivate = reinterpret_cast<LPNMITEMACTIVATE>(pNMHDR);
	// TODO: 在此添加控件通知处理程序代码
	POINT pt = {};
	GetCursorPos(&pt);
	HMENU hMenu = LoadMenu(AfxGetApp()->m_hInstance, MAKEINTRESOURCE(IDR_MENU1));
	hMenu = GetSubMenu(hMenu, 0);
	TrackPopupMenu(hMenu, TPM_CENTERALIGN, pt.x, pt.y, 0, m_hWnd, NULL);
	*pResult = 0;
}


void C任务管理器Dlg::OnProcessList()
{
	// TODO: 在此添加命令处理程序代码
	m_ProcessList.DeleteAllItems();
	InitProcessList();
}


void C任务管理器Dlg::OnKillProcess()
{
	// TODO: 在此添加命令处理程序代码
	DWORD dwIndex = (DWORD)m_ProcessList.GetFirstSelectedItemPosition();
	dwIndex--;
	CString ProcssID = m_ProcessList.GetItemText(dwIndex, 1);
	DWORD nProcessID = _ttoi(ProcssID);
	HANDLE hProcess =  OpenProcess(PROCESS_ALL_ACCESS, NULL, nProcessID);
	TerminateProcess(hProcess, 1);


}


void C任务管理器Dlg::OnCreateProcess()
{
	// TODO: 在此添加命令处理程序代码
	CFileDialog file(TRUE, L"exe", L"*.exe", OFN_OVERWRITEPROMPT | OFN_HIDEREADONLY, L"可执行文件|*.exe|所有文件|.*", NULL);
	if (file.DoModal() == IDOK)
	{
		CString FilePath = file.GetPathName();
		STARTUPINFO si = {};
		si.cb = sizeof(STARTUPINFO);
		PROCESS_INFORMATION pi = {};
		CreateProcess(FilePath, NULL, NULL, NULL, NULL, CREATE_NEW_CONSOLE, NULL, NULL, &si, &pi);
	}


}

相关文章:

  • 内存泄漏指什么?常见的内存泄漏有哪些?
  • SpringBoot整合Swagger
  • 电子电气架构 --- 主机厂电子电气架构演进
  • QT:QPen、QBrush、与图形抗锯齿的关联
  • 线程同步辅助类的使用
  • 4.3MISC流量分析练习-wireshark-https
  • 内网渗透测试-Vulnerable Docker靶场
  • 深入探究OPA1612AIDR:性能剖析、引脚功能、应用实例与注意事项
  • 如何让别人的电脑蓝屏?(没有任何实质性损害,重启后仍然能正常运行;可恶搞)
  • 向量数据库milvus部署
  • Baklib驱动内容中台智能推荐优化
  • 《深度剖析:生成对抗网络中生成器与判别器的高效协作之道》
  • 华为数通Datacom认证体系详解:从HCIA到HCIE的进阶路径
  • 两种常见视频传输线材
  • 如何防止 Instagram 账号被盗用:安全设置与注意事项
  • 学习threejs,Materials常量汇总
  • Linux中Shell运行原理和权限(下)(4)
  • 玄机-第六章 流量特征分析-蚁剑流量分析
  • 软件工程应试复习(考试折磨版)
  • C语言机试编程题
  • 新华时评:中国维护国际经贸秩序的立场坚定不移
  • 人民财评:网售“婴儿高跟鞋”?不能让畸形审美侵蚀孩子身心
  • 视频丨雄姿英发!中国仪仗队步入莫斯科红场
  • 中方是否认同俄方关于新纳粹主义观点?外交部:联大曾多次通过相关决议
  • 上海杨浦:优秀“博主”购房最高可获200万补贴
  • 长三角地区中华老字号品牌景气指数发布,哪些牌子是你熟悉的?