C# 重定向stdin和stdout,其中stdin首先关闭
这实际上与我已经回答过的另一个问题有关。这个问题是: 我的问题是(我认为)获取输入的程序应该在程序输出之前退出。下面是我正在做的bash等价物:tccat-I/dev/sr0-t1 | ffmpeg-I--r1-t1-s96x72-ss5/tmp/wsmanage/preview\u tmp/test\%03d.jpg 这只是使用一个名为tccat的程序来读取dvd的第一个标题。这将输出到ffmpeg,ffmpeg将以每秒1帧的速度创建一个输出文件,该文件的.jpg格式为1秒长。一旦输出帧(或两帧),它就存在。这个很好用 但是,下面的代码没有。我得到了“将行打印到ffmpeg”的SCAD,而命令行版本只需一秒钟就可以退出。然后,它会在30或40秒后停止打印。FFmpeg永远不会退出,我的程序也永远不会继续 我做得对吗C# 重定向stdin和stdout,其中stdin首先关闭,c#,ipc,C#,Ipc,这实际上与我已经回答过的另一个问题有关。这个问题是: 我的问题是(我认为)获取输入的程序应该在程序输出之前退出。下面是我正在做的bash等价物:tccat-I/dev/sr0-t1 | ffmpeg-I--r1-t1-s96x72-ss5/tmp/wsmanage/preview\u tmp/test\%03d.jpg 这只是使用一个名为tccat的程序来读取dvd的第一个标题。这将输出到ffmpeg,ffmpeg将以每秒1帧的速度创建一个输出文件,该文件的.jpg格式为1秒长。一旦输出帧(或两
Process tccatProcess = new Process();
tccatProcess.StartInfo.FileName = "tccat";
tccatProcess.StartInfo.Arguments = String.Format("-i {0} -T {1}", devNode, title);
tccatProcess.StartInfo.UseShellExecute = false;
tccatProcess.StartInfo.RedirectStandardOutput = true;
Process ffmpegProcess = new Process();
string bashSafePreviewTemplate = slasher.bashSlash(previewTempDir + "/test%03d.jpg");
ffmpegProcess.StartInfo.FileName = "ffmpeg";
ffmpegProcess.StartInfo.Arguments = String.Format("-i - -r 1 -t 1 -s {1}x{2} -ss {3} {0}",
bashSafePreviewTemplate, width, height, timePosition);
ffmpegProcess.StartInfo.UseShellExecute = false;
ffmpegProcess.StartInfo.RedirectStandardInput = true;
try{
tccatProcess.Start();
ffmpegProcess.Start();
StreamReader tccatOutput = tccatProcess.StandardOutput;
StreamWriter ffmpegInput = ffmpegProcess.StandardInput;
string line;
while(!ffmpegProcess.HasExited)
{
ffmpegProcess.Refresh();
if((line = tccatOutput.ReadLine()) != null)
{
Console.WriteLine("Printing line to ffmpeg");
Console.Out.Flush();
ffmpegInput.WriteLine(line);
ffmpegInput.Flush();
}
}
Console.WriteLine("Closing tccat");
Console.Out.Flush();
tccatProcess.Close();
Console.WriteLine("Tccat closed");
Console.Out.Flush();
}catch(Exception e){
//uninteresting log code
return false;
}
由于您正在处理视频和图像,tccat不输出二进制数据吗?如果是这样的话,您不应该直接读/写输入/输出流,而不是将它们包装在文本读取器中吗
如果您想为任何想做类似事情的人使用,以下是新版本,其中应用了KeeperOfTheSoul的建议:
Process tccatProcess = new Process();
tccatProcess.StartInfo.FileName = "tccat";
tccatProcess.StartInfo.Arguments = String.Format("-i {0} -T {1}", devNode, title);
tccatProcess.StartInfo.UseShellExecute = false;
tccatProcess.StartInfo.RedirectStandardOutput = true;
Process ffmpegProcess = new Process();
string bashSafePreviewTemplate = slasher.bashSlash(previewTempDir + "/test%03d.jpg");
ffmpegProcess.StartInfo.FileName = "ffmpeg";
ffmpegProcess.StartInfo.Arguments = String.Format("-i - -r 1 -t 1 -s {1}x{2} -ss {3} {0}",
bashSafePreviewTemplate, width, height, timePosition);
ffmpegProcess.StartInfo.UseShellExecute = false;
ffmpegProcess.StartInfo.RedirectStandardInput = true;
Console.WriteLine("tccat command: {0} {1}", tccatProcess.StartInfo.FileName, tccatProcess.StartInfo.Arguments);
Console.WriteLine("ffmpeg command: {0} {1}", ffmpegProcess.StartInfo.FileName, ffmpegProcess.StartInfo.Arguments);
//return true;
try{
tccatProcess.Start();
ffmpegProcess.Start();
BinaryReader tccatOutput = new BinaryReader(tccatProcess.StandardOutput.BaseStream);
BinaryWriter ffmpegInput = new BinaryWriter(ffmpegProcess.StandardInput.BaseStream);
int buffSize = 4096;
byte[] buff = new byte[buffSize];
while(!ffmpegProcess.HasExited)
{
ffmpegProcess.Refresh();
buff = tccatOutput.ReadBytes(buffSize);
ffmpegInput.Write(buff);
ffmpegInput.Flush();
}
tccatProcess.Kill();
tccatProcess.Close();
ffmpegProcess.Close();
}catch(Exception e){
//uninteresting log code
return false;
}
啊,这可能是我的问题。我没有意识到streamreader/streamwriter不是二进制安全的。我会尽快尝试这个,并将结果发回!也谢谢你的链接。我知道我需要做什么。即使这不是我唯一的问题,我可以看出这确实是需要解决的问题。