第三方库生成任意大小的16位带符号立体声样本的顺序缓冲区。我无法从这些缓冲区中找出要使用的框架/功能。我一直在使用AudioQueue处理this answer中的示例,但是它在我无法解析的方式中显然是不完整的(使用局部变量就好像它们是对象成员,未声明的变量等)。

Apple的SpeakHere示例项目中的代码仅显示了如何使用AudioQueue播放文件中的音频。有人可以指出我正确的方向吗?

最佳答案

弄清楚了。这是一个使用从基于窗口的应用程序项目模板创建的AudioQueue的简单,完整且有效的白噪声发生器。

StaticAppDelegate.h:

//  StaticAppDelegate.h

#import <UIKit/UIKit.h>

// STATIC ADDITIONS
// Add > Existing Frameworks... > AudioToolbox.framework to your Target
#import <AudioToolbox/AudioToolbox.h>
// END STATIC ADDITIONS

@interface StaticAppDelegate : NSObject <UIApplicationDelegate> {
    UIWindow *window;
}

@property (nonatomic, retain) IBOutlet UIWindow *window;

// STATIC ADDITIONS
- (void)startStatic;
- (void)handleBufferCompleteForQueue:(AudioQueueRef)inAQ
                              buffer:(AudioQueueBufferRef)inBuffer;
// END STATIC ADDITIONS

@end


StaticAppDelegate.m:

//  StaticAppDelegate.m

#import "StaticAppDelegate.h"

// STATIC ADDITIONS
#define kNumberBuffers 4
#define kBufferSize 2048

// AudioQueue callback
void AQOutputCallback(
                      void                 *inData,
                      AudioQueueRef        inAQ,
                      AudioQueueBufferRef  inBuffer
                      )
{
    StaticAppDelegate *staticApp = (StaticAppDelegate *)inData;
    [staticApp handleBufferCompleteForQueue:inAQ buffer:inBuffer];
}
// END STATIC ADDITIONS

@implementation StaticAppDelegate

@synthesize window;


- (void)applicationDidFinishLaunching:(UIApplication *)application {

    // Override point for customization after application launch
    [window makeKeyAndVisible];

    // STATIC ADDITIONS
    [self startStatic];
    // END STATIC ADDITIONS
}

- (void)dealloc {
    [window release];
    [super dealloc];
}

// STATIC ADDITIONS
- (void)startStatic
{
    srandom(time(NULL));

    // initiate audio session
    AudioSessionInitialize(NULL, NULL, NULL, NULL);
    UInt32 category = kAudioSessionCategory_MediaPlayback; // plays through sleep lock and silent switch
    AudioSessionSetProperty(kAudioSessionProperty_AudioCategory, sizeof(category), &category);
    AudioSessionSetActive(true);

    // following is modified from http://stackoverflow.com/questions/1710133/playing-generated-audio-on-an-iphone

    // setup queue
    AudioQueueRef audioQueue;
    AudioQueueBufferRef buffers[kNumberBuffers];
    AudioStreamBasicDescription format;
    memset(&format, 0, sizeof(format));
    format.mSampleRate          = 44100;
    format.mFormatID            = kAudioFormatLinearPCM;
    format.mFormatFlags         = kLinearPCMFormatFlagIsSignedInteger | kLinearPCMFormatFlagIsPacked;
    format.mChannelsPerFrame    = 1;
    format.mBitsPerChannel      = 16;
    format.mBytesPerFrame       = (format.mBitsPerChannel / 8) * format.mChannelsPerFrame;
    format.mFramesPerPacket     = 1;
    format.mBytesPerPacket      = format.mBytesPerFrame * format.mFramesPerPacket;

    AudioQueueNewOutput(&format,
                        AQOutputCallback,
                        self,
                        CFRunLoopGetCurrent(),
                        kCFRunLoopCommonModes,
                        0,
                        &audioQueue);

    // allocate and fill the buffers
    for (int i = 0; i < kNumberBuffers; ++i)
    {
        AudioQueueAllocateBuffer(audioQueue, kBufferSize, &buffers[i]);
        AQOutputCallback(self, audioQueue, buffers[i]);
    }

    AudioQueueSetParameter(audioQueue, kAudioQueueParam_Volume, 1.0);
    AudioQueueStart(audioQueue, NULL);
}

- (void)handleBufferCompleteForQueue:(AudioQueueRef)inAQ
                              buffer:(AudioQueueBufferRef)inBuffer
{
    inBuffer->mAudioDataByteSize = inBuffer->mAudioDataBytesCapacity;

    int *buffer = (int *)inBuffer->mAudioData;
    for (int i = 0; i < inBuffer->mAudioDataByteSize/sizeof(int); ++i)
    {
        buffer[i] = (int)rand(); // refill the buffer
    }
    AudioQueueEnqueueBuffer(inAQ, inBuffer, 0, NULL);
}
// END STATIC ADDITIONS

@end


一旦我弄清楚如何真正地重新填充缓冲区(使用某种类型转换的inBuffer->mAudioData),其他所有内容都就位了。

10-07 18:09