1

我有一个产生多个线程的应用程序,其中一个线程运行一个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。知道为什么会这样吗?

4

1 回答 1

0

开始输出读线

链接描述了如何在示例中完全按照您的要求进行操作。

为什么不使用内置的 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() 后,您还需要 Close()

于 2016-02-18T20:03:46.213 回答