iOS 使用AUGraph录音同时播放(并转码成Mp3)

如果只需要完成简单的录音功能,苹果有更高级、方便的接口供开发者使用:AVAudioRecorder,面向对象,不关心细节实现。但是如果开发者想要拿到实时数据并对其进行处理,就要用到Audio Unit Services和Audio Processing Graph Services。下面我会介绍如何使用它们来完成一个最简单的录音DEMO。

AudioSession

首先,我们需要了解下AudioSession这个类。先看下苹果对它的介绍:

iOS handles audio behavior at the app, inter-app, and device levels through audio sessions

iOS通过AudioSession来控制APP中的音频表现,跨应用和硬件设备。按我的理解,它就是用来设定最基础的音频配置的,比如:
1、当耳机被拔出,是否停止音频的播放?
2、本APP的音频播放是否和其他APP的音频实现混音?还是让其他APP的音频暂停?
3、是否允许APP获取麦克风数据?

在本文中,我们需要用到麦克风录音并且播放,调用以下代码,APP就会弹出窗口询问是否允许APP访问麦克风:

AVAudioSession *audioSession = [AVAudioSession sharedInstance];
[audioSession setCategory:AVAudioSessionCategoryPlayAndRecord error:&error];

Audio Processing Graph

先用一张表格(来自苹果文档)介绍下Audio Unit的类型:

Purpose Audio units
Effect iPod Equalizer
Mixing 3D Mixer
Multichannel Mixer
I/O Remote I/O
Voice-Processing I/O
Generic Output
Format conversion Format Converter

7种类型,4个作用:均衡器、混合、输入/输出和格式转换。
在本DEMO中只使用了Remote I/O完成简单的录音和播放功能。
而Audio Unit并不能独立完成工作,需要配合AUGraph来使用。AUGraph是管理者,不同的Unit作为Node添加到AUGraph中去发挥作用,如下图AUGraph管理着Mixer Unit和Remote I/O Unit:


AudioProcessingGraphBeforeEQ_2x.png

声明一个Remote I/O类型的Node,并添加到AUGraph中:

AUNode remoteIONode;
AudioComponentDescription componentDesc; //关于Node的描述
componentDesc.componentType = kAudioUnitType_Output;
componentDesc.componentSubType = kAudioUnitSubType_RemoteIO;
componentDesc.componentManufacturer = kAudioUnitManufacturer_Apple;
componentDesc.componentFlags = 0;
componentDesc.componentFlagsMask = 0;

CheckError(NewAUGraph(&auGraph),"couldn't NewAUGraph"); //创建AUGraph
CheckError(AUGraphOpen(auGraph),"couldn't AUGraphOpen"); //打开AUGraph

CheckError(AUGraphAddNode(auGraph,&componentDesc,&remoteIONode),"couldn't add remote io node");
CheckError(AUGraphNodeInfo(auGraph,remoteIONode,NULL,&remoteIOUnit),"couldn't get remote io unit from node");

Remote I/O Unit

Remote I/O Unit 属于Audio Unit其中之一,是一个与硬件设备相关的Unit,它分为输入端和输出端,如扬声器、麦克风和耳机等。在这里我们需要在录音的同时播放,所以我们要让输入端和输出端的Unit相连,如下图所示:


IO_unit_2x.png

其中Element0代表着输出端,Element1代表输入端;而每个Element又分为Input scope和Output scope。我们具体要做的就是将Element0的Output scope和喇叭接上,Element1的Intput和麦克风接上。代码如下:

 UInt32 oneFlag = 1; 
 CheckError(AudioUnitSetProperty(remoteIOUnit,
                                    kAudioOutputUnitProperty_EnableIO,
                                    kAudioUnitScope_Output,
                                    kOutputBus,
                                    &oneFlag,
                                    sizeof(oneFlag)),"couldn't kAudioOutputUnitProperty_EnableIO with kAudioUnitScope_Output");
    
 CheckError(AudioUnitSetProperty(remoteIOUnit,
                                    kAudioOutputUnitProperty_EnableIO,
                                    kAudioUnitScope_Input,
                                    kInputBus,
                                    &oneFlag,
                                    sizeof(oneFlag)),"couldn't kAudioOutputUnitProperty_EnableIO with kAudioUnitScope_Input");

然后设置一下输入输出的音频格式:

AudioStreamBasicDescription mAudioFormat;
mAudioFormat.mSampleRate         = 44100.0;//采样率
mAudioFormat.mFormatID           = kAudioFormatLinearPCM;//PCM采样
mAudioFormat.mFormatFlags        = kAudioFormatFlagIsSignedInteger | kAudioFormatFlagIsPacked;
mAudioFormat.mFramesPerPacket    = 1;//每个数据包多少帧 mAudioFormat.mChannelsPerFrame   = 1;//1单声道,2立体声
mAudioFormat.mBitsPerChannel     = 16;//语音每采样点占用位数
mAudioFormat.mBytesPerFrame      = mAudioFormat.mBitsPerChannel*mAudioFormat.mChannelsPerFrame/8;//每帧的bytes数
mAudioFormat.mBytesPerPacket     = mAudioFormat.mBytesPerFrame*mAudioFormat.mFramesPerPacket;//每个数据包的bytes总数,每帧的bytes数*每个数据包的帧数
mAudioFormat.mReserved           = 0;
UInt32 size = sizeof(mAudioFormat);
    
CheckError(AudioUnitSetProperty(remoteIOUnit,
                                    kAudioUnitProperty_StreamFormat,
                                    kAudioUnitScope_Output,
                                    1,
                                    &mAudioFormat,
                                    size),"couldn't set kAudioUnitProperty_StreamFormat with kAudioUnitScope_Output");
    
CheckError(AudioUnitSetProperty(remoteIOUnit,
                                    kAudioUnitProperty_StreamFormat,
                                    kAudioUnitScope_Input,
                                    0,
                                    &mAudioFormat,
                                    size),"couldn't set kAudioUnitProperty_StreamFormat with kAudioUnitScope_Input");

至此我们就差不多完成了,只差最后一个步骤:设置CallBack,每次音频从麦克风进来转化为数字信号时就会调用此CallBack函数,将数字信号作你所想要的处理,完了再送到输出端去进行播放。回调函数是一个C语言的静态方法,代码如下:

static OSStatus CallBack(
                            void      *inRefCon,
                            AudioUnitRenderActionFlags  *ioActionFlags,
                            const AudioTimeStamp   *inTimeStamp,
                            UInt32       inBusNumber,
                            UInt32       inNumberFrames,
                            AudioBufferList    *ioData)
{
    RecordTool *THIS=(__bridge RecordTool*)inRefCon;
    
    OSStatus renderErr = AudioUnitRender(THIS->remoteIOUnit, ioActionFlag,  inTimeStamp, 1, inNumberFrames, ioData);

//--------------------------------------------//
//              在这里处理音频数据               //
//--------------------------------------------//
    
//转Mp3,后面有时间再详细介绍下    
//     [THIS->cover convertPcmToMp3:ioData->mBuffers[0] toPath:THIS->outPath];

    
    return renderErr;
}
ioData->mBuffers[n] //n=0~1,单声道n=0,双声道n=1
ioData->mBuffers[0].mData //PCM数据
ioData->mBuffers[0].mDataByteSize //PCM数据的长度

定义好CallBack函数后,将其与AUGraph关联起来:

AURenderCallbackStruct inputProc;
inputProc.inputProc = CallBack;
inputProc.inputProcRefCon = (__bridge void *)(self);
CheckError(AUGraphSetNodeInputCallback(auGraph, remoteIONode, 0, &inputProc),"Error setting io output callback");
 
CheckError(AUGraphInitialize(auGraph),"couldn't AUGraphInitialize" );
CheckError(AUGraphUpdate(auGraph, NULL),"couldn't AUGraphUpdate" );

//最后再调用以下代码即可开始录音
CheckError(AUGraphStart(auGraph),"couldn't AUGraphStart");
CAShow(auGraph);

最后

因为时间的关系,未能细说PCM转MP3(使用LAME)的内容,我已经把DEMO上传到GitHub,有需要的朋友可以下载来看看,往后有时间了我再补全关于转码MP3的内容。
如果觉得我的DEMO对您有帮助,请Star,非常感谢!

更新

使用LAME转码请看iOS-使用Lame转码:PCM->MP3

关于我

目前在职iOS开发,业余时间独立开发App,现有上架作品:Mini记账
公z号:沙拉可乐 分享独立开发的干货和背后的故事

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,142评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,298评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,068评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,081评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,099评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,071评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,990评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,832评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,274评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,488评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,649评论 1 347
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,378评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,979评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,625评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,796评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,643评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,545评论 2 352

推荐阅读更多精彩内容