开始一个过程 -
ProcessStartInfo psi = new ProcessStartInfo("G:\\SampleWinApp.exe");
psi.UseShellExecute = false;
psi.CreateNoWindow = true;
Process prcs = Process.Start(psi);
使用PostMessage发送WM_CLOSE
const int WM_CLOSE = 0x0010;
public void SendCloseSignal(Process proc)
{
uint uiPid = (uint) proc.Id;
bool bResult = EnumWindows(new WNDENUMPROC(EnumWindowsProc), uiPid);
if (!bResult && Marshal.GetLastWin32Error() == 0) {
object objWnd = processWnd[uiPid];
if (objWnd != null) {
IntPtr ptrWnd = (IntPtr) objWnd;
PostMessage(ptrWnd, WM_CLOSE, 0, 0);
return;
}
}
foreach (ProcessThread thread in proc.Threads) {
PostThreadMessage((uint) thread.Id, WM_CLOSE, UIntPtr.Zero, IntPtr.Zero);
}
}
private static bool EnumWindowsProc(IntPtr hwnd, uint lParam)
{
uint uiPid = 0;
if (GetParent(hwnd) == IntPtr.Zero)
{
GetWindowThreadProcessId(hwnd, ref uiPid);
if (uiPid == lParam)
{
processWnd[uiPid] = hwnd;
return false;
}
}
return true;
}
使用CreateNoWindow = false
启动exe,发送WM_CLOSE消息并正常关闭应用程序。
使用CreateNoWindow = true
时,WM_CLOSE消息永远不会达到。甚至PostThreadMessage似乎也不起作用。有没有办法发送WM_CLOSE消息?我找了一天找到解决方案......没有运气。
编辑:为每个应用程序安装Windows服务。启动/停止服务可启动/停止应用程序。目前我们在服务停止时终止应用程序。由于它的强力杀戮,应用程序不会优雅地死亡。一些应用程序会监听CTRL signals。现在,我只需要一些方法将WM_CLOSE消息发送到这些应用程序。
编辑2:如果有窗口,WM_CLOSE会触发CTRL_CLOSE_EVENT。但是当使用CreateNoWindow = true启动任何进程时,这永远不会触发。
答案 0 :(得分:3)
有没有办法发送WM_CLOSE消息?
WM_CLOSE
被发送到一个窗口。如果进程中没有窗口,则没有任何内容可以处理该消息。如果您希望关闭没有窗口的进程,则发送WM_CLOSE
不是解决方案。
看起来你只想杀死进程。当控制台进程具有关联的窗口时,您正在使用WM_CLOSE
消息来触发CTRL_CLOSE_EVENT
信号。
由于CTRL_CLOSE_EVENT
已经是一种相当残酷的方法来杀死一个进程,在我看来,你完全可以完全理所当然地杀死它。您已有Process
个对象。只需使用Process.Kill()
杀死它。
另一方面,也许您真正想问的问题是:
如何在另一个过程中发出
CTRL_CLOSE_EVENT
信号?
相关的问题可以在这里找到:Can I send a ctrl-C (SIGINT) to an application on Windows?我认为最好的答案是Nemo1024,而不是被接受的答案。该答案链接到他的博客文章:http://stanislavs.org/stopping-command-line-applications-programatically-with-ctrl-c-events-from-net/
据我了解,该文章应该为您提供解决问题所需的信息。它列出了四种情景。前两个涉及WM_CLOSE
作为触发关闭信号的手段。第三是杀死这个过程。第四个可能就是你要找的东西。当然,第四种选择是最复杂的。引用文章:
在这个帖子的底部找到一个提示后,我终于咬了一下子弹并开始了没有窗口的过程。当停止进程的时候,父进程将其控制台附加到子进程,停止自己侦听Ctrl-C事件并将事件发送到控制台,以便附加的子进程终止。然后它为自己恢复Ctrl-C处理程序并释放控制台。