播放完毕后调用 AVAudioPlayerNode 上的 stop() 会导致崩溃

dma*_*200 3 audio xcode swift avaudioplayernode

我有一个 AVAudioPlayerNode 对象,它在播放完毕后发送回调。该回调会触发应用程序中的许多其他功能,其中之一会发出一条礼貌的 stop() 消息。由于某种原因,在 AVAudioPlayerNode 完成时调用 stop() 会导致崩溃。在此处的代码中,我对其进行了缩写,以便 AVAudioPlayerNode 仅立即调用 stop 来演示效果(而不是包括我的整个应用程序)。你可以清楚地看到它崩溃了。我不明白为什么。a) 节点仍在播放并且 stop() 停止它,或者 b) 节点播放完毕并且 stop 可以被忽略。

我的猜测是,这是某种边缘情况,它位于文件缓冲区的最末尾,并且处于不明确的状态,没有剩余的缓冲区,但技术上仍在播放。也许调用 stop() 尝试刷新剩余的缓冲区,但它们是空的?

func testAVAudioPlayerNode(){
    let testBundle = Bundle(for: type(of: self))
    let url = URL(fileURLWithPath: testBundle.path(forResource: "19_lyrics_1", ofType: "m4a")!)
    let player = AVAudioPlayerNode()
    let engine = AVAudioEngine()
    let format = engine.mainMixerNode.outputFormat(forBus: 0)
    engine.attach(player)
    engine.connect(player, to: engine.mainMixerNode, format: format)
    let delegate = FakePlayMonitorDelegate()
    do {
        let audioFile = try AVAudioFile(forReading: url)
        let length = audioFile.length
        player.scheduleFile(audioFile, at: nil, completionCallbackType: AVAudioPlayerNodeCompletionCallbackType.dataPlayedBack, completionHandler: {(completionType) in
            print("playing = \(player.isPlaying)")
            player.stop()
        })
        try engine.start()
        let expectation = self.expectation(description: "playback")
        delegate.expectation = expectation
        player.play()
        self.waitForExpectations(timeout: 6.0, handler: nil)
    } catch {

    }
}
Run Code Online (Sandbox Code Playgroud)

Ore*_*est 6

就我而言,.stop()从主线程调用该方法(您可以使用另一个)已经解决了问题。

DispatchQueue.main.async {
    player.stop()
}
Run Code Online (Sandbox Code Playgroud)

这可能是僵局。我注意到它completionHandler是从后台队列调用的,这是一个串行队列,我们​​称其为“渲染队列”。似乎该.stop()方法正在尝试在“渲染队列”上做一些工作,但目前“渲染队列”正忙于completionHandler.

死锁截图