Windows编程----结束进程

进程有启动就有终止,通过CreateProcess函数可以启动一个新的子进程,但是如何终结子进程呢?主要有四种方法:

通过主线程的入口函数(main函数、WinMain函数)的return关键字终止进程

一个应用程序只有一个入口函数,对于控制台来说是main函数,对于GUI程序来说这个入口函数一般是WinMain。入口函数通过return关键字返回或者程序自上而下执行完成之后,进程可以自动终止,进程相关的所有资源都会被操作清理。这也是开发中最常用的进程终止方法,也是强烈推荐大家使用的终止进程的方式。

那么如何能获取到子进程的返回值呢,以下面代码为例,这是一个最简单的控制台程序,它返回-3。

复制代码
#include <iostream>
#include <Windows.h>
int main()
{
	return -3;
}

首先演示在cmd控制台或者bat脚本中如何获取该进程的最终返回值:通过echo %errorlevel%命令可以获取到上一个cmd命令的返回值。如下:

在程序中获取子进程的返回值,CreateProcess函数创建子进程成功之后,主进程可以拿到子进程的句柄。通过调用GetExitCodeProcess函数可以获取到某个进程的返回值,函数原型如下:

复制代码
BOOL GetExitCodeProcess(
  HANDLE  hProcess,//子进程句柄
  LPDWORD lpExitCode//用于接受子进程的返回值
);

请看以下代码:

复制代码
#include <iostream>
#include <Windows.h>
int main()
{
	//即将启动的exe程序路径
	LPCWSTR lpApplicationName = L"D:\\project\\ConsoleApp1\\x64\\Debug\\NewApp.exe";

	// 定义启动信息和进程信息结构
	STARTUPINFOW si;
	PROCESS_INFORMATION pi;

	// 初始化启动信息结构
	ZeroMemory(&si, sizeof(si));
	si.cb = sizeof(si);
	ZeroMemory(&pi, sizeof(pi));

	BOOL ret = CreateProcess(lpApplicationName,
		NULL,
		NULL,
		NULL,
		FALSE,
		CREATE_NEW_CONSOLE,
		NULL,
		NULL,
		&si,
		&pi);
	// 等待进程结束
	WaitForSingleObject(pi.hProcess, INFINITE);

	// 获取子进程的返回值
	DWORD exitCode;
	GetExitCodeProcess(pi.hProcess, &exitCode);
	// 将无符号整数转换为有符号整数
	int signedExitCode = static_cast<int>(exitCode);
	std::cout << "Child process exited with code: " << signedExitCode << std::endl;
	return 0;
}

主程序在创建子进程之后,首先调用WaitForSingleObject等待子进程结果,然后调用GetExitCodeProcess函数获取子进程的返回值。但是这里注意,如果子进程返回0或者正数是没问题的,因为DWORD是一个无符号的整数类型,但是如果子进程返回负数的话,就需要我们手动将无符号整数转换为有符号整数。最后执行结果如下,可以看到正常获取到了子进程的返回值。

通过ExitProcess函数终止进程

ExitProcess函数可以终止进程,并且会设置一个进程的返回值。该函数原型只有一个参数,就是进程的返回值:

复制代码
void ExitProcess(
  UINT uExitCode //进程返回值
);

注意:调用ExitProcess之后,进程会直接退出,ExitProcess之后的函数都不会继续执行,在程序中,一定要尽量避免直接调用这个函数,我们以下面的代码为例:

复制代码
#include <iostream>
#include <Windows.h>
int main(int argc, char** argv)
{
	ExitProcess(-4);
	std::cout << "Hello World!" << std::endl;
	return -3;
}

当我们双击编译好的控制台程序之后,会发现hello World!并没有被执行,并且成功的返回了-4,效果如下:

通过TerminateProcess函数终止进程

我们首先来修改子进程程序,通过一个无限while循环使进程永远不会退出:

复制代码
#include <iostream>
#include <Windows.h>
int main(int argc, char** argv)
{
	while (true) {
		std::cout << "Hello World!" << std::endl;
		Sleep(1000);
	}
}

然后按照上面的代码,创建完子进程之后,我们可以调用TerminateProcess函数直接终止子进程。

复制代码
	BOOL ret = CreateProcess(lpApplicationName,
		NULL,
		NULL,
		NULL,
		FALSE,
		CREATE_NEW_CONSOLE,
		NULL,
		NULL,
		&si,
		&pi);

	TerminateProcess(pi.hProcess, -3);

TerminateProcess函数非常的简单粗暴,会强制将子进程终结,也应该尽量避免直接调用该函数终止进程,只有在其他办法都无法强制终结子进程的情况下,才可以使用该函数强制终结进程。

相关推荐
初願致夕霞12 分钟前
Linux_进程
linux·c++
Thera77744 分钟前
【Linux C++】彻底解决僵尸进程:waitpid(WNOHANG) 与 SA_NOCLDWAIT
linux·服务器·c++
Wei&Yan1 小时前
数据结构——顺序表(静/动态代码实现)
数据结构·c++·算法·visual studio code
呉師傅1 小时前
【使用技巧】Adobe Photoshop 2024调整缩放与布局125%后出现点菜单项漂移问题的简单处理
运维·服务器·windows·adobe·电脑·photoshop
wregjru1 小时前
【QT】4.QWidget控件(2)
c++
浅念-1 小时前
C++入门(2)
开发语言·c++·经验分享·笔记·学习
小羊不会打字1 小时前
CANN 生态中的跨框架兼容桥梁:`onnx-adapter` 项目实现无缝模型迁移
c++·深度学习
Max_uuc1 小时前
【C++ 硬核】打破嵌入式 STL 禁忌:利用 std::pmr 在“栈”上运行 std::vector
开发语言·jvm·c++
近津薪荼1 小时前
dfs专题4——二叉树的深搜(验证二叉搜索树)
c++·学习·算法·深度优先
艾莉丝努力练剑2 小时前
【Linux:文件】Ext系列文件系统(初阶)
大数据·linux·运维·服务器·c++·人工智能·算法