1
 OSStatus err = AudioQueueNewOutput(&audioDescription, AudioPlayerAQOutputCallback, ( void* )self, nil, nil, 0, &audioQueue);
    if( err != noErr )
        NSLog(@"Couldn't open AudioFile.");

    err = AudioQueueAddPropertyListener(audioQueue, kAudioQueueProperty_IsRunning, isRunningProc, self);
    if( err != noErr )
        NSLog(@"Couldn't register for playback state changes.");

此回调函数仅在 AudioQueueStart(audioQueue, NULL) 之后调用一次;

我叫什么 AudioQueuePause(audioQueue);

或音频到达终点。

static void isRunningProc(void *              inUserData,
                      AudioQueueRef           inAQ,
                      AudioQueuePropertyID    inID)

我错过了什么?

4

1 回答 1

1

我对此做了一个简短的测试:

当您恢复暂停时,它确实看起来像回调,它既不是为了暂停也不是为了开始而被调用。

但这不是你无法解决的。你以某种方式开始了这首歌。这将触发属性监听器。同样,如果歌曲停止。或者你阻止它。您可能必须在播放例程中以某种方式自己触发属性侦听器:

if (bytesRead == 0) {
    //This will trigger the property listener
    AudioQueueStop(inAQ, false);
}
else {
    AudioQueueEnqueueBuffer(inAQ, inBuffer, 0, NULL);
}

就 AudioQueue 而言,只要您继续向它提供要播放的音频缓冲区,它就会继续播放。(我还测试了根本不提供任何缓冲区,这不会触发停止,因此您必须显式调用停止来触发属性侦听器。)

这意味着您已经知道您的歌曲是否正在播放。通过单击您的按钮请求暂停或取消暂停。如果歌曲没有播放,不要做任何事情。如果歌曲正在播放,请调用 AudioQueuePause 并设置一个您已暂停音乐的标志。记得检查错误代码。(见下文(1))。如果标志表明您已暂停音乐,请调用 AudioQueueStart,并清除表明您是否已暂停的标志。再次检查错误代码。

(1) 为什么要检查错误代码?

首先,虽然不太可能,但可能会发生错误,因为它是蓝月亮。

但是,我担心的是多个线程。AudioQueue 显然在与您的 GUI 不同的线程上运行。这意味着如果您测试一个标志是否正在播放音乐,则该状态不能完全可信,因为它可能在您测试状态后发生了变化。另一个线程可能已经潜入您的测试和基于该测试的操作之间。

假设您检查歌曲是否已经在播放。(是的。)然后您要求歌曲暂停,但歌曲实际上已停止,因为在您要求歌曲暂停之前它已到达结尾。然后你要求暂停这首歌。但它已经停止了。

那会发生什么?我真的不知道。在这种情况下,它甚至可能不是问题,但这样的事情值得考虑。它需要测试,或者至少需要查阅文档。

另一个场景怎么样?如果歌曲停止了,而您要求重新开始,该怎么办。我认为这是一个更糟糕的情况,但它可能不是问题。再次考虑这些情况并检查文档,甚至测试自己。

于 2013-12-01T15:24:44.687 回答