在 Process.StandardOutput.Read() 方法之前是不是需要睡眠?

Posted

技术标签:

【中文标题】在 Process.StandardOutput.Read() 方法之前是不是需要睡眠?【英文标题】:Is sleep is required before Process.StandardOutput.Read() method?在 Process.StandardOutput.Read() 方法之前是否需要睡眠? 【发布时间】:2012-12-13 19:33:44 【问题描述】:

我正在尝试在 C# 程序(例如 cmd)中创建一个子进程并使用进程 IO 流执行读/写。我正在使用 StandardOutput.Read() 方法来读取进程输出。

当我在 Read() 之前放置一个 Thread.Sleep() 方法时,它会提供完整的输出,但如果我删除它,它只会显示单行输出。

代码如下:

string sProcess = "cmd.exe";
ProcessStartInfo psiInfo = new ProcessStartInfo();
psiInfo.FileName = sProcess;
psiInfo.CreateNoWindow = true;
psiInfo.UseShellExecute = false;
psiInfo.RedirectStandardOutput = true;
psiInfo.RedirectStandardError = true;
psiInfo.RedirectStandardInput = true;
Process pChild = new Process();
pChild.StartInfo = psiInfo;
if (pChild.Start())

    int ch;
    do
    
        Thread.Sleep(50);
        ch = pChild.StandardOutput.Peek();
        if (ch > 0)
            Console.Write((char)pChild.StandardOutput.Read());
     while (ch > 0);
    Console.WriteLine("exit");
    pChild.StandardInput.WriteLine("exit");

启用睡眠的输出:

Microsoft Windows [版本 6.1.7601] 版权所有 (c) 2009 Microsoft Corporation。版权所有。 D:\ProcessDemo_001\bin\Release>退出

禁用睡眠的输出:

Microsoft Windows [版本 6.1.7601] 退出

我想知道为什么会这样?

【问题讨论】:

如果你只想要所有的输出,你应该使用ReadToEnd 我使用了 ReadToEnd,但由于我也在读取输出后向进程发送输入,因此 ReadToEnd 功能阻塞,我只得到一个空白屏幕。你能解决这个问题吗。我会很感激的。不过谢谢你的建议…… @Servy 使用cmd.exe,它永远不会回来,因为它等待输入。 这似乎是问题所在。没有理由使用cmd.exe,他应该直接在可执行文件上启动进程 【参考方案1】:

我想知道为什么会这样?

您的循环运行速度快于生成的输出。一旦它通过输出,它就结束了,所以它永远不会看到第二行。

【讨论】:

那么除了使用睡眠之外还有其他方法吗,因为它只会减慢处理速度...... @liquidcoder 如果您正在执行一个无需输入即可运行到完成的进程,那么您可以等待它退出或读取到流的末尾。如果没有,您可能需要阅读,直到获得“提示”字符(在这种情况下,您可以阅读直到阅读'>',然后输入... 谢谢!我会做的。还有一件事,有什么方法可以让我们提前知道流程的输出流中有多少数据可用?【参考方案2】:

您的 Peek 命令可能在 pChild 输出文本之前执行。发生这种情况时,ch 将为 0,while 循环退出。

【讨论】:

是的。我猜你打对了。但是我也尝试过使用 Read 函数,然后只要进程没有可用的输出,do--while 循环就会无限循环,因为 Read 块... 您可以尝试使用 yield() 而不是陡峭的() 这会导致线程停止处理并告诉操作系统进行上下文切换。

以上是关于在 Process.StandardOutput.Read() 方法之前是不是需要睡眠?的主要内容,如果未能解决你的问题,请参考以下文章

process.StandardOutput.ReadToEnd() 假死

没有获得进程的输出 (SFC.exe /VERIFYONLY)。 Process.StandardOutPut.ReadToEnd 提供输出,其中 /n/0/n 包含在各种空间中

通过 C# 的 Process 类生成时,处理标准错误和程序输出的正确方法?

分配的变量引用在哪里,在堆栈中还是在堆中?

NOIP 2015 & SDOI 2016 Round1 & CTSC 2016 & SDOI2016 Round2游记

秋的潇洒在啥?在啥在啥?