我无法弄清楚为什么这个代码在与后台工作程序一起使用时失败了

时间:2013-03-26 22:20:47

标签: c# audio backgroundworker

我正在使用Alvas Audio库将文件从Mp3转换为Wave格式,以便我可以编辑它们,然后将它们转换回Mp3。转换为Wave工作正常,但当我尝试转换回Mp3时,我遇到了麻烦。出于某种原因,这种失败与我使用后台工作程序执行初始转换的事实有关。

我知道如果没有图书馆的源代码,就很难尝试弄清楚发生了什么,但我希望有人可能会对可能出现的问题提出建议。

当我在没有后台工作程序的情况下同步调用此相同代码时,它可以正常工作。任何想法?

以下是我从后台工作人员调用以执行转换的代码:

    public Tuple<float, float> convertMp3ToWav(Track track) //with Detection, duration check, and TODO: silence removal
    {
        try
        {
            string baseFile = Path.GetFileName(track.location);
            ////////////////////////////
            //string baseFile = track.location.Remove(track.location.Length - 4);
            string outputFile = directory + "Temp\\" + baseFile.Remove(baseFile.Length - 4) + ".wav";
            cleanupFiles.Add(outputFile);
            if (!File.Exists(outputFile))
            {
                int soundStart = -1;
                int soundEnd = 0;
                Mp3Reader mr = new Mp3Reader(File.OpenRead(track.location));
                IntPtr mp3Format = mr.ReadFormat();
                IntPtr pcmFormat = AudioCompressionManager.GetCompatibleFormat(mp3Format, AudioCompressionManager.PcmFormatTag);
                AcmConverter acm = new AcmConverter(mp3Format, pcmFormat, false);

                int sec = 1024;
                int i = 0;

                bool soundFound = false;
                while (true)
                {
                    byte[] mp3Data = mr.ReadDataInBytes(i, sec);
                    if (mp3Data.Length == 0)
                    {
                        break;
                    }
                    byte[] pcmData = acm.Convert(mp3Data);
                    foreach (byte d in pcmData) //THIS SECTION CHECKS IF THE Section in question has silence
                    {
                        if (d != 0)
                        {
                            soundFound = true;
                        }

                    }
                    if ((soundStart == -1) && (soundFound == true)) //if no beginning sound has been found yet, but has now been found
                    {
                        soundStart = i; //This will be precise to whatever value of sec has been chosen
                    }
                    else if ((soundStart != -1) && (soundFound == true)) //this is a possible end value
                    {
                        soundEnd = i;   //this value will be rewritten each time there is sound found after soundstart is set.
                                        //so this value will remain the same if no further sound is found in the track, and will represent the 
                                        //end of sound in the track
                    }

                    soundFound = false;
                    i += sec;
                }
                int finalDuration = soundEnd - soundStart;
                mr.Close();
                Mp3Reader reader = new Mp3Reader(File.OpenRead(track.location));
                IntPtr thisFormat = reader.ReadFormat();
                byte[] completeTrack = reader.ReadDataInBytes(soundStart, finalDuration);
                byte[] convertedTrack = AudioCompressionManager.Convert(thisFormat, pcmFormat, completeTrack, false);
                WaveWriter ww = new WaveWriter(File.OpenWrite(outputFile), AudioCompressionManager.FormatBytes(pcmFormat));
                ww.WriteData(convertedTrack);
                ww.Close();
                reader.Close();

                float bpm = performBeatDetection(track);
                float duration = getTrackDuration(track);

                return new Tuple<float, float>(bpm, duration);
            }
            else
            {
                //File already exists, just remove silence, get bpm and duration
                //string silenceRemoved = removeSilenceFromTrack(outputFile);
                float bpm = performBeatDetection(track);
                float duration = getTrackDuration(track);
                return new Tuple<float, float>(bpm, duration);
            }
        }
        catch (Alvas.Audio.AudioException e)
        {
            MessageBox.Show("ERROR: " + e.ToString());
            return new Tuple<float, float>(0f, 0f);
        }
    }

编辑:

特定故障是库的一个例外。当您想使用Alvas将文件从一种音频格式转换为另一种音频格式时,首先要读取文件当前格式的格式,如您所见

IntPtr mp3Format = mr.ReadFormat();

然后转换,你调用静态方法

AudioCompressionManager.GetCompatibleFormat(oldFormat, newFormatTag);

当我在使用后台工作程序后调用最后一个方法时,它会以错误的英语引发异常“它没有成功进行格式转换”。在库中使用稍微不同的方法:

AudioCompressionManager.GetCompatibleFormatList(oldFormat);

揭示当我不使用后台工作器,并且我使用后一种方法(GetCompatibleFormatList)时,它返回51个结果,其中一个是转换为我想要的mp3格式。

如果我在使用后台工作程序执行初始转换后执行相同的方法,它只返回20个结果,其中没有一个是兼容的mp3格式。

1 个答案:

答案 0 :(得分:0)

当你没有定义“麻烦”时,很难具体说明。但也许这些想法会有所帮助:

  • 后台工作程序使用的任何数据(或文件)是否被前台线程触及? (例如track参数在被使用时是否可以以任何方式处理或修改?)

  • 前台线程在后台线程运行时是否调用任何库API?您的任何代码是否在多个线程上调用库代码?您的前台线程是否正确等待后台线程完成结果? (例如,作为测试尝试只是让你的前台线程加入后台线程(或者做一个忙碌的等待/休眠),这样它只是在处理执行时等待 - 问题会消失吗?)

  • 库是否会执行任何可能需要在特定线程上调用的内容 - 特别是,是否需要在UI线程中运行以更新进度条或类似的内容? (这包括在代码中调用可能需要调用UI线程的事件处理程序)

  • 您传回主线程的返回类型以外是否有其他数据?如果是这样,你使用同步(锁或易失性等)来确保你读取正确的(最新的)值?

  • 在处理方法返回之前,您确定所有数据都已写入,刷新到磁盘,所有文件都已关闭吗?