杀死C#进程优雅地无法正常工作

时间:2016-02-18 18:14:48

标签: c# .net process iperf

我有一个应用程序产生多个线程,其中一个运行iPerf可执行文件,用于监视网络可靠性。此过程将无限期运行,直到用户尝试关闭窗口。这就是问题所在。我试图优雅地关闭该进程,以便iPerf服务器不会挂起,但我似乎无法使其正常工作。如果我从命令提示符手动运行命令并按Ctrl+c,我可以将其关闭,但这似乎不是通过编程方式轻松完成的。

我尝试了多项内容,包括process.Kill();process.StandardInput.Close()甚至process.StandardInput.WriteLine("\x3");,但这些似乎都没有向流程发送优雅的关闭消息。 process.Kill();导致服务器挂起或无法在下次启动,而其他两个选项根本不会停止服务器。但是手册ctrl+c工作得很好。

以下是我的代码片段:

iperf_proc = new Process();
iperf_proc.StartInfo.FileName = Application.StartupPath + ".\\iperf3.exe";
String argumentStr = " -c " + test_data.host + " -t 0";
iperf_proc.StartInfo.Arguments = argumentStr;
iperf_proc.StartInfo.UseShellExecute = false;
iperf_proc.StartInfo.RedirectStandardOutput = true;
iperf_proc.StartInfo.RedirectStandardInput = true;
iperf_proc.StartInfo.RedirectStandardError = true;
iperf_proc.Start();
iperfRunning = true;
iperf_proc.BeginOutputReadLine();
iperf_proc.BeginErrorReadLine();
while (false == iperf_proc.HasExited)
{
    if (true == processCancelled)
    {
        iperf_proc.StandardInput.Close(); // Doesn't Work!
        iperf_proc.StandardInput.WriteLine("\x3"); // Doesn't Work!
        iperf_proc.StandardInput.Flush(); // Doesn't Work!
    }
}
iperf_proc.WaitForExit();

非常感谢任何帮助。谢谢!

更新

根据Hans在评论中提出的建议,我尝试在代码中添加一些内容以便发送ctrl+c事件。

[DllImport("kernel32.dll")]
[return: MarshalAs(UnmanagedType.Bool)]
static extern bool GenerateConsoleCtrlEvent(uint dwCtrlEvent, uint dwProcessGroupId);
private enum CtrlEvents
{
     CTRL_C_EVENT = 0,
     CTRL_BREAK_EVENT = 1
}
private void closeBtn_Click(object sender, EventArgs e)
{
     processCancelled = true;
     //iperf_proc.CloseMainWindow();
     bool succeeded = GenerateConsoleCtrlEvent((uint)CtrlEvents.CTRL_C_EVENT, (uint)iperf_proc.Id);
}

这根本不起作用。该进程仍在运行,并且添加的函数返回false。我确实检查了传递的进程ID是否与任务管理器中进程的id匹配。一切都很好,但GenerateConsoleCtrlEvent函数返回false。知道为什么会这样吗?

1 个答案:

答案 0 :(得分:0)

BeginOutputReadLine

链接描述了如何在示例中完成您正在寻找的内容。

为什么不使用内置的Exited事件来等待你没有阻止?特别是如果你正在产生多个线程。如果你想阻止那么WaitForExit()也可用。

if (true == processCancelled)
{
    iperf_proc.StandardInput.Close(); // Doesn't Work!
    iperf_proc.StandardInput.WriteLine("\x3"); // Doesn't Work!
    iperf_proc.StandardInput.Flush(); // Doesn't Work!
}

如果您关闭标准输入,您将如何写/刷它?

在你有WaitForExit()后,你还需要关闭()