我使用ffmpeg工具开发了一个用于视频音频合并的wpf程序。我正在使用后台工作者类来实现此目的。我会在这里包含我的代码。
public VideoAudioMergeUserControl()
{
InitializeComponent();
bw.WorkerSupportsCancellation = true;
bw.WorkerReportsProgress = true;
bw.DoWork += new DoWorkEventHandler(bw_DoWork);
bw.RunWorkerCompleted += new RunWorkerCompletedEventHandler(bw_RunWorkerCompleted);
bw.ProgressChanged += new ProgressChangedEventHandler(bw_ProgressChanged);
}
void bw_ProgressChanged(object sender, ProgressChangedEventArgs e)
{
string progress= e.ProgressPercentage.ToString();
}
void bw_RunWorkerCompleted(object sender, RunWorkerCompletedEventArgs e)
{
}
void bw_DoWork(object sender, DoWorkEventArgs e)
{
Process interactiveProcess = new Process();
string processOutput = "";
interactiveProcess.StartInfo.FileName = @"E:\ffmpeg.exe";
interactiveProcess.StartInfo.UseShellExecute = false;
interactiveProcess.StartInfo.CreateNoWindow = true;
interactiveProcess.StartInfo.RedirectStandardInput = true;
interactiveProcess.StartInfo.RedirectStandardError = true;
ObservableCollection<VideoAudioCombined> videosAudios = (ObservableCollection<VideoAudioCombined>)e.Argument;
foreach (VideoAudioCombined va in videosAudios)
{
string absoluteVideoPath = aviVideoPath + va.Video;
string absoluteAudioPath = wavAudioPath + va.Audio;
string relativeVideoPathSplit = va.Video.Substring(0, va.Video.Length - 4);
string relativeAudioPathSplit = va.Audio.Substring(0, va.Audio.Length - 4);
string outputavi = relativeVideoPathSplit + "_" + relativeAudioPathSplit + ".avi";
string SelectedFileNameToSave = selectedFolder + outputavi;
interactiveProcess.StartInfo.Arguments = "-i " + absoluteVideoPath + " -i " + absoluteAudioPath + " -c:v copy -c:a aac -strict experimental " + SelectedFileNameToSave + " ";
interactiveProcess.Start();
interactiveProcess.EnableRaisingEvents = true;
interactiveProcess.ErrorDataReceived += new DataReceivedEventHandler((process, outputEventArgs) => processOutput += outputEventArgs.Data);
interactiveProcess.BeginErrorReadLine();
bw.ReportProgress(100);
}
}
private void ButtonMerge_Click(object sender, RoutedEventArgs e)
{
System.Windows.Forms.FolderBrowserDialog folderDlg = new System.Windows.Forms.FolderBrowserDialog();
folderDlg.ShowNewFolderButton = true;
folderDlg.Description = Properties.Resources.MSG_Select_Folder_Merge_Audio_Video;
// Show the FolderBrowserDialog.
System.Windows.Forms.DialogResult result = folderDlg.ShowDialog();
if (result == System.Windows.Forms.DialogResult.OK)
{
selectedFolder = folderDlg.SelectedPath+"\\";
}
bw.RunWorkerAsync(VideoAudioCollection);
}
循环的第一次迭代不会产生任何问题。第二次迭代带来了以下问题 - “已经在流上启动了异步读取操作”。请帮忙。
答案 0 :(得分:0)
在启动后台工作程序之前,请确保IsBusy
为false:
if (!bw.IsBusy)
{
bw.RunWorkerAsync(VideoAudioCollection);
}
如果IsBusy是真的,那意味着后台工作者仍在工作,但还没有完成它的功能。在这种情况下,它会抛出异常。如果发生这种情况,您可以显示一个消息框说明 “仍在进行合并操作,请等到完成后再进行”。
您也可以尝试取消当前正在运行的操作
bw.CancelAsync();
答案 1 :(得分:0)
错误的原因是行:interactiveProcess.BeginErrorReadLine();
您不必在foreach循环中使用单个进程调用该方法。
在第一次迭代中,您启动了异步读取操作并继续。实际上,在下一次迭代中,此操作已经开始。 根据文件:
当异步读取操作开始时,将调用事件处理程序 每次关联的进程将一行文本写入其中时 StandardError流。
您必须查看启动该过程的逻辑。我认为你应该为循环的每次迭代开始一个不同的过程,因为,对于每次迭代,你使用的是具有不同参数的相同过程,但是相同的StandardError
流。
有关方法here的更多信息。