如何终止CreateProcess()创建的进程?

dig*_*jay 17 c++ winapi mfc createprocess visual-c++

我创建了一个使用过程CreateProcess().这是代码:

STARTUPINFO si = {0};
PROCESS_INFORMATION pi = {0};
result = CreateProcess("C:\\AP\\DatabaseBase\\dbntsrv.exe", NULL, NULL, NULL, FALSE, 0, NULL, "C:\\ADP\\SQLBase", &si, &pi)
Run Code Online (Sandbox Code Playgroud)

如何获取此特定过程的Handle和processId?并最终用它来关闭这个过程?
谢谢.

Har*_*der 12

在结构中pi你得到:

typedef struct _PROCESS_INFORMATION {
    HANDLE hProcess;
    HANDLE hThread;
    DWORD  dwProcessId;
    DWORD  dwThreadId;
} PROCESS_INFORMATION, *LPPROCESS_INFORMATION;
Run Code Online (Sandbox Code Playgroud)

第一个参数是进程的句柄.

您可以使用该句柄来结束该过程:

BOOL WINAPI TerminateProcess(
    __in  HANDLE hProcess,
    __in  UINT uExitCode
);
Run Code Online (Sandbox Code Playgroud)

hProcess [in]
要终止的进程的句柄.

句柄必须具有PROCESS_TERMINATE访问权限.有关更多信息,请参阅进程安全性和访问权限.

uExitCode [in]
进程使用的退出代码以及由于此调用而终止的线程.使用GetExitCodeProcess函数检索进程的退出值.使用GetExitCodeThread函数检索线程的退出值.


hmj*_*mjd 11

PROCESS_INFORMATION结构pi变量中返回进程的句柄.

了TerminateProcess()函数可以被用来终止进程.但是,您应该考虑为什么需要终止进程以及无法正常关闭的原因.

注意你需要在调用之前设置cb成员:siCreateProcess()

si.cb = sizeof(STARTUPINFO);
Run Code Online (Sandbox Code Playgroud)

编辑:

禁止控制台窗口指定CREATE_NO_WINDOW,作为调用中的创建标志(第六个参数)CreateProcess().

编辑(2):

要禁止窗口,请在调用之前尝试设置STARTUPINFO结构的以下成员CreateProcess():

STARTUPINFO si = {0};
si.cb          = sizeof(STARTUPINFO);
si.dwFlags     = STARTF_USESHOWWINDOW;
si.wShowWindow = FALSE;
Run Code Online (Sandbox Code Playgroud)


vll*_*vll 5

干净地关闭进程

要干净地关闭进程,您应该首先发送关闭信号:

如何在 Win32 中“干净地”终止应用程序。

如果您绝对必须关闭某个进程,请执行以下步骤:

  1. 向要关闭的进程拥有的所有顶级窗口发布 WM_CLOSE。许多 Windows 应用程序通过关闭来响应此消息。

注意:控制台应用程序对 WM_CLOSE 的响应取决于它是否安装了控制处理程序。

使用 EnumWindows() 查找目标窗口的句柄。在您的回调函数中,检查 Windows 的进程 ID 是否与您要关闭的进程匹配。您可以通过调用 GetWindowThreadProcessId() 来完成此操作。建立匹配后,使用 PostMessage() 或 SendMessageTimeout() 将 WM_CLOSE 消息发布到窗口。

  1. 使用 WaitForSingleObject() 等待进程的句柄。确保使用超时值等待,因为在很多情况下 WM_CLOSE 不会关闭应用程序。请记住使超时时间足够长(使用 WaitForSingleObject() 或使用 SendMessageTimeout()),以便用户可以响应为响应 WM_CLOSE 消息而创建的任何对话框。

  2. 如果返回值是 WAIT_OBJECT_0,则应用程序将自己彻底关闭。如果返回值为 WAIT_TIMEOUT,则必须使用 TerminateProcess() 关闭应用程序。

注意:如果您从 WaitForSingleObject() 获得除 WAIT_OBJECT_0 或 WAIT_TIMEOUT 之外的返回值,请使用 GetLastError() 确定原因。

通过遵循这些步骤,您可以给应用程序最好的机会彻底关闭(除了 IPC 或用户干预)。

请参阅此答案以获取代码。

终止进程

如果您不关心干净关机,则可以使用TerminateProcess(). 但是,重要的是要注意它TerminateProcess()是异步的;它启动终止并立即返回。如果必须确保进程已终止,请WaitForSingleObject()使用进程句柄调用该函数。

注意:访问权限 PROCESS_TERMINATESYNCHRONIZE是必需的。

TerminateProcess(pi.hProcess, 0);

// 500 ms timeout; use INFINITE for no timeout
const DWORD result = WaitForSingleObject(pi.hProcess, 500);
if (result == WAIT_OBJECT_0) {
    // Success
}
else {
    // Timed out or an error occurred
}

CloseHandle(pi.hProcess);
CloseHandle(pi.hThread);
Run Code Online (Sandbox Code Playgroud)

不关闭,等完成

如果进程将自行完成,您可以等待它完成而不是终止。

WaitForSingleObject(pi.hProcess, INFINITE);

CloseHandle(pi.hProcess);
CloseHandle(pi.hThread);
Run Code Online (Sandbox Code Playgroud)