使用不同的起点条目快速混合两个音频文件

Posted

技术标签:

【中文标题】使用不同的起点条目快速混合两个音频文件【英文标题】:Mix two audio files in swift with different start point entry 【发布时间】:2018-11-16 15:48:52 【问题描述】:

我正在尝试在 swift AVFoundations 中混合两个音频文件。我有一个混合两个音频并同时播放它们的解决方案。我真正想要的是在一段时间后开始第二个音频。例如,音频 1 播放,然后第二个音频在 10(给定时间)秒后开始播放。任何帮助表示赞赏。提前致谢。

private var audioFiles: Array<String>
private var audioEngine: AVAudioEngine = AVAudioEngine()
private var mixer: AVAudioMixerNode = AVAudioMixerNode()

func Play() 
    // do work in a background thread
    DispatchQueue.global(qos: .background).async 
      self.audioEngine.attach(self.mixer)
      self.audioEngine.connect(self.mixer, to: self.audioEngine.outputNode, format: nil)
      // !important - start the engine *before* setting up the player nodes
      try! self.audioEngine.start()

      let fileManager = FileManager.default
      for audioFile in self.audioFiles 
        // Create and attach the audioPlayer node for this file
        let audioPlayer = AVAudioPlayerNode()
        self.audioEngine.attach(audioPlayer)
        // Notice the output is the mixer in this case
        self.audioEngine.connect(audioPlayer, to: self.mixer, format: nil)
        let fileUrl = NSURL.init(fileURLWithPath: fileName.removingPercentEncoding!)
        var file : AVAudioFile

        // We should probably check if the file exists here ¯\_(ツ)_/¯
        try! AVAudioFile.init(forReading: fileUrl.absoluteURL!)

        audioPlayer.scheduleFile(file, at: nil, completionHandler: nil)
        audioPlayer.play(at: nil)
      
    
  

【问题讨论】:

【参考方案1】:

你应该使用scheduleSegment。

func scheduleWithOffset(_ offset: TimeInterval) 

    let samplerate1 = file1.processingFormat.sampleRate
    player1.scheduleSegment(file1,
                            startingFrame: 0,
                            frameCount: AVAudioFrameCount(file1.length),
                            at: AVAudioTime(sampleTime: 0, atRate: samplerate1))

    let samplerate2 = file2.processingFormat.sampleRate
    player2.scheduleSegment(file2,
                            startingFrame: 0,
                            frameCount: AVAudioFrameCount(file2.length),
                            at: AVAudioTime(sampleTime: AVAudioFramePosition(offset * samplerate2), atRate: samplerate2))

    //This can take an indeterminate amount of time, so both files should be prepared before either starts.
    player1.prepare(withFrameCount: 8192)
    player2.prepare(withFrameCount: 8192)

    // Start the files at common time slightly in the future to ensure a synchronous start.
    let hostTimeNow = mach_absolute_time()
    let hostTimeFuture = hostTimeNow + AVAudioTime.hostTime(forSeconds: 0.2);
    let startTime = AVAudioTime(hostTime: hostTimeFuture)

    player1.play(at: startTime)
    player2.play(at: startTime)

【讨论】:

【参考方案2】:

我个人没有使用AVAudioEngineAVAudioMixerNode 的经验,但是,您可以使用Timer 来做到这一点。

您当前拥有audioPlayer.play(at: nil),但是,它始终会在建立此播放器后立即播放。我会做或尝试做的是为此应用某种计时器。

var timeAfter = 10
var timer = NSTimer.scheduledTimerWithInterval(timeAfter, target: self, selector: #selector(playAudio), userInfo: audioPlayer, repeats: false)

将取代

audioPlayer.play(at: nil)

然后你会添加定时器的功能来播放音频。

func playAudio(timer:NSTimer)
  var audioPlayer = timer.userInfo as AVAudioPlayerNode
  audioPlayer.play()

【讨论】:

【参考方案3】:

你可以添加另一个 playerNode 到混音器,然后像这样延迟播放:

    let audioPlayer2 = AVAudioPlayerNode()
    self.audioEngine.attach(audioPlayer2)
    self.audioEngine.connect(audioPlayer2, to: self.mixer, format: nil)
    var file2 : AVAudioFile = try! AVAudioFile.init(forReading: fileUrl2!)
    func delayTime(_ delayTime : TimeInterval) -> AVAudioTime
        let  outputFormat = audioPlayer2.outputFormat(forBus: 0)
        let  startSampleTime = (audioPlayer2.lastRenderTime ?? AVAudioTime()).sampleTime +  Int64(Double(delayTime) * outputFormat.sampleRate);
        return  AVAudioTime(sampleTime: startSampleTime, atRate: outputFormat.sampleRate)
    
    audioPlayer2.scheduleFile(file2, at: nil, completionHandler: nil)
    audioPlayer2.play(at: delayTime(3.0))

您还可以在 node2 和混音器之间挂钩 sampleDelay AU。这让事情变得更容易,无需手动编程。

【讨论】:

以上是关于使用不同的起点条目快速混合两个音频文件的主要内容,如果未能解决你的问题,请参考以下文章

使用 NAudio 混合两个音频文件

如何通过在 iphone 中混合多个音频来录制音频?

修剪音频返回空文件(Python)

在 IOS 上使用 AVComposition 混合两个音频文件

使用扩展音频文件服务将两个文件混合在一起

iOS - 混合两个音频文件