2

我正在尝试使用 StreamReader 从进程中读取输出数据,但 StreamReader 阻塞并且不会返回任何输出。

我的流程如下所示:

ProcessStartInfo startInfo = new ProcessStartInfo();
startInfo.Arguments = args;
startInfo.FileName = filename;
StartInfo.WorkingDirectory = aDirectory;
StartInfo.UseShellExecute = false;
StartInfo.RedirectStandardOutput = true;
Process p = new Process();
p.StartInfo = startInfo;
p.Start();

之后立即调用 StreamReader:

StreamReader strmRead = p.StandardOutput;
char[] output = new char[4096];
while(true){
   strmRead.Read(output,0,output.Length);
   string outputString = new string(output);
   Debug.WriteLine(outputString);
}

代码在调用 Read 方法时挂起。当我手动终止程序时,进程的输出被写入调试控制台。进程输出不使用换行符,因此使用 Process.OutputDataReceived 不起作用。如何从流中获取进程输出而不让它无限期地阻塞?

编辑:鉴于我已经得到的答案,这似乎是进程没有放弃标准输出或不刷新输出的问题,而不是我的代码有任何问题。如果其他人有任何见解,请随时发表评论。

4

3 回答 3

1

你可以这样做:

String outputString = strmRead.ReadToEnd();
于 2012-05-31T18:32:01.893 回答
1

您正在读取 4096 字节并且可能更少,因此流块。

此外,还有更有效的方式从流中读取文本。 TextReaderReadLine方法,试试吧。

http://msdn.microsoft.com/en-us/library/system.io.textreader.readline.aspx

顺便说一句,while (true)???你打算如何退出?

于 2012-05-31T18:28:46.027 回答
1

我知道这个问题很老了。但我有一个类似的问题。我尝试使用 Peek() 方法,但即使 Peek() 返回 -1,它也不总是流的结尾。

我通过启动一个新线程来解决我的问题,该线程在 Peek() 返回 -1 时尝试读取下一个字符。

                    string toRead = "";
                    do
                    {
                        if (reader.Peek() == -1)
                        {
                            Thread td = new Thread(TryReading);
                            td.Start();
                            Thread.Sleep(400);
                            if (ReadSuccess == false)
                            {
                                try
                                {
                                    td.Abort();
                                }
                                catch (Exception ex) { }
                                break;
                            }
                            else
                            {
                                toRead += ReadChar;
                                ReadSuccess = false;
                            } 
                        }
                        toRead += (char)reader.Read();
                    } while (true);

TryReading() 方法在这里定义:

static char ReadChar = 'a';
static bool ReadSuccess = false;

static void TryReading(object callback)
{
    int read = reader.Read();
    ReadChar = (char)read;
    ReadSuccess = true;
}

基本上......如果线程花费太长时间来读取字符 - 我中止它并使用它到目前为止读取的文本。

这解决了我的问题。

于 2014-07-27T09:50:53.883 回答