ios 音訊佇列實現錄音和播音(轉)
轉自 http://www.verydemo.com/demo_c92_i301380.html
使用AudioQueue來實現音訊播放功能時最主要的步驟,可以更簡練的歸納如下。
1. 開啟播放音訊檔案
2. 取得播放音訊檔案的資料格式
3. 準備播放用的佇列
4. 將緩衝中的資料移動到佇列中
5. 開始播放
6. 在回撥函式中進行佇列處理
1.playAudio.h
聲明瞭一個Objective-C類
1.playAudio.h
聲明瞭一個Objective-C類
1.播放音樂
// // playAudio.h // ffmpegPlayAudio // // Created by infomedia xuanyuanchen on 12-3-26. // Copyright (c) 2012年 xuanyuanchen. All rights reserved. // #import #import #import #define NUM_BUFFERS 3 @interface playAudio : NSObject{ //播放音訊檔案ID AudioFileID audioFile; //音訊流描述物件 AudioStreamBasicDescription dataFormat; //音訊佇列 AudioQueueRef queue; SInt64 packetIndex; UInt32 numPacketsToRead; UInt32 bufferByteSize; AudioStreamPacketDescription *packetDescs; AudioQueueBufferRef buffers[NUM_BUFFERS]; } //定義佇列為例項屬性 @property AudioQueueRef queue; //播放方法定義 -(id)initWithAudio:(NSString *) path; //定義快取資料讀取方法 -(void) audioQueueOutputWithQueue:(AudioQueueRef)audioQueue queueBuffer:(AudioQueueBufferRef)audioQueueBuffer; -(UInt32)readPacketsIntoBuffer:(AudioQueueBufferRef)buffer; //定義回撥(Callback)函式 static void BufferCallack(void *inUserData,AudioQueueRef inAQ, AudioQueueBufferRef buffer); @end
2.playAudio.m
playAudio的實現
// // playAudio.m // ffmpegPlayAudio // // Created by infomedia infomedia on 12-3-26. // Copyright (c) 2012年 infomedia. All rights reserved. // #import "playAudio.h"
//實際測試中發現,這個gBufferSizeBytes=0x10000;對於壓縮的音訊格式(mp3/aac等)沒有任何問題,但是如果輸入的音訊檔案格式是wav,會出現只播放幾秒便暫停的現象;而手機又不可能去申請更大的記憶體去處理wav檔案,不知到大家能有什麼好的方法給點建議
static UInt32 gBufferSizeBytes=0x10000;//It muse be pow(2,x)
@implementation playAudio @synthesize queue; //回撥函式(Callback)的實現 static void BufferCallback(void *inUserData,AudioQueueRef inAQ, AudioQueueBufferRef buffer){ playAudio* player=(__bridge playAudio*)inUserData; [player audioQueueOutputWithQueue:inAQ queueBuffer:buffer]; } //快取資料讀取方法的實現 -(void) audioQueueOutputWithQueue:(AudioQueueRef)audioQueue queueBuffer:(AudioQueueBufferRef)audioQueueBuffer{ OSStatus status; //讀取包資料 UInt32 numBytes; UInt32 numPackets=numPacketsToRead; status = AudioFileReadPackets(audioFile, NO, &numBytes, packetDescs, packetIndex,&numPackets, audioQueueBuffer->mAudioData); //成功讀取時 if (numPackets>0) { //將緩衝的容量設定為與讀取的音訊資料一樣大小(確保記憶體空間) audioQueueBuffer->mAudioDataByteSize=numBytes; //完成給佇列配置快取的處理 status = AudioQueueEnqueueBuffer(audioQueue, audioQueueBuffer, numPackets, packetDescs); //移動包的位置 packetIndex += numPackets; } } //音訊播放的初始化、實現
//在ViewController中宣告一個PlayAudio物件,並用下面的方法來初始化
//self.audio=[[playAudioalloc]initWithAudio:@"/Users/xuanyuanchen/audio/daolang.mp3"];
-(id) initWithAudio:(NSString *)path{ if (!(self=[super init])) return nil; UInt32 size,maxPacketSize; char *cookie; int i; OSStatus status; //開啟音訊檔案 status=AudioFileOpenURL((CFURLRef)[NSURL fileURLWithPath:path], kAudioFileReadPermission, 0, &audioFile); if (status != noErr) { //錯誤處理 NSLog(@"*** Error *** PlayAudio - play:Path: could not open audio file. Path given was: %@", path); return nil; } for (int i=0; i音訊資料格式 size = sizeof(dataFormat); AudioFileGetProperty(audioFile, kAudioFilePropertyDataFormat, &size, &dataFormat); //建立播放用的音訊佇列 AudioQueueNewOutput(&dataFormat, BufferCallback, self, nil, nil, 0, &queue); //計算單位時間包含的包數 if (dataFormat.mBytesPerPacket==0 || dataFormat.mFramesPerPacket==0) { size=sizeof(maxPacketSize); AudioFileGetProperty(audioFile, kAudioFilePropertyPacketSizeUpperBound, &size, &maxPacketSize); if (maxPacketSize > gBufferSizeBytes) { maxPacketSize= gBufferSizeBytes; } //算出單位時間內含有的包數 numPacketsToRead = gBufferSizeBytes/maxPacketSize; packetDescs=malloc(sizeof(AudioStreamPacketDescription)*numPacketsToRead); }else { numPacketsToRead= gBufferSizeBytes/dataFormat.mBytesPerPacket; packetDescs=nil; } //設定Magic Cookie,參見第二十七章的相關介紹 AudioFileGetProperty(audioFile, kAudioFilePropertyMagicCookieData, &size, nil); if (size >0) { cookie=malloc(sizeof(char)*size); AudioFileGetProperty(audioFile, kAudioFilePropertyMagicCookieData, &size, cookie); AudioQueueSetProperty(queue, kAudioQueueProperty_MagicCookie, cookie, size); } //建立並分配緩衝空間 packetIndex=0; for (i=0; i佇列處理開始,此後系統開始自動呼叫回撥(Callback)函式 AudioQueueStart(queue, nil); return self; } -(UInt32)readPacketsIntoBuffer:(AudioQueueBufferRef)buffer { UInt32 numBytes,numPackets; //從檔案中接受資料並儲存到快取(buffer)中 numPackets = numPacketsToRead; AudioFileReadPackets(audioFile, NO, &numBytes, packetDescs, packetIndex, &numPackets, buffer->mAudioData); if(numPackets >0){ buffer->mAudioDataByteSize=numBytes; AudioQueueEnqueueBuffer(queue, buffer, (packetDescs ? numPackets : 0), packetDescs); packetIndex += numPackets; } else{ return 1;//意味著我們沒有讀到任何的包 } return 0;//0代表正常的退出 } @end
2.錄音
(1.)record.h
static const int kNumberBuffers = 3;#import
#import
#import
@interface Record : NSObject
{
AudioStreamBasicDescription mDataFormat;
AudioQueueRef mQueue;
AudioQueueBufferRef mBuffers[kNumberBuffers];
UInt32 bufferByteSize;
AVAudioPlayer *player;
}
@property(nonatomic,assign) bool mIsRunning;
@property(nonatomic,retain) NSMutableData* recordData;
-(void)record;
-(void)stop;
-(void)play;
-(void)pause;
-(void)dealWithData:(NSData*)data;
@end
(2.)record.m
#import "Record.h"
@implementation Record@synthesize mIsRunning = _mIsRunning;
@synthesize recordData =_recordData;
-(id)init
{
self = [super init];
if (self) {
AudioSessionInitialize(NULL, NULL, NULL, (__bridge void*)self);
// self.recordData =[NSMutableData data];
self.mIsRunning=false;
}
return self;
}
static void HandleInputBuffer (
void *inUserData,
AudioQueueRef inAQ,
AudioQueueBufferRef inBuffer,
const AudioTimeStamp *inStartTime,
UInt32 inNumPackets,
const AudioStreamPacketDescription *inPacketDesc
) {
Record *recorderPro = (__bridge Record *)inUserData;
if (inNumPackets > 0 && recorderPro.mIsRunning){
int pcmSize = inBuffer->mAudioDataByteSize;
char *pcmData = (char *)inBuffer->mAudioData;
NSData *data = [[NSData alloc] initWithBytes:pcmData length:pcmSize];
[recorderPro dealWithData:data];
// [recorderPro.recordData appendData:data];
AudioQueueEnqueueBuffer(inAQ, inBuffer, 0, NULL);
}
}
void DeriveBufferSize (
AudioQueueRef audioQueue,
AudioStreamBasicDescription &ASBDescription,
Float64 seconds,
UInt32 *outBufferSize
) {
static const int maxBufferSize = 0x50000;
int maxPacketSize = ASBDescription.mBytesPerPacket;
if (maxPacketSize == 0) {
UInt32 maxVBRPacketSize = sizeof(maxPacketSize);
AudioQueueGetProperty (
audioQueue,
kAudioQueueProperty_MaximumOutputPacketSize,
&maxPacketSize,
&maxVBRPacketSize
);
}
Float64 numBytesForTime =
ASBDescription.mSampleRate * maxPacketSize * seconds;
*outBufferSize =
UInt32 (numBytesForTime < maxBufferSize ?
numBytesForTime : maxBufferSize);
}
-(void)dealWithData:(NSData*)data
{
[self changeVoice:data];
}
-(void)changeVoice:(NSData*)audioData
{
soundtouch::SoundTouch mSoundTouch;
mSoundTouch.setSampleRate(16000);
mSoundTouch.setChannels(1);
mSoundTouch.setTempoChange(0.05);
mSoundTouch.setPitchSemiTones(12);
mSoundTouch.setRateChange(-0.7);
mSoundTouch.setSetting(SETTING_SEQUENCE_MS, 40);
mSoundTouch.setSetting(SETTING_SEEKWINDOW_MS, 16);
mSoundTouch.setSetting(SETTING_OVERLAP_MS, 8);
if (audioData != nil) {
char *pcmData = (char *)audioData.bytes;
int pcmSize = audioData.length;
int nSamples = pcmSize / 2;
mSoundTouch.putSamples((short *)pcmData, nSamples);
相關推薦
ios 音訊佇列實現錄音和播音(轉)
轉自 http://www.verydemo.com/demo_c92_i301380.html 使用AudioQueue來實現音訊播放功能時最主要的步驟,可以更簡練的歸納如下。 1. 開啟播放音訊檔案 2. 取得播放音訊檔案的資料格式 3. 準備播放用的佇列
Java for Web學習筆記(九十):訊息和叢集(5)利用websocket實現訂閱和釋出(上)
叢集中的訂閱和釋出 利用spring framework在本app內的訂閱和釋出十分簡單。當我們系統越來越複雜的時候,我們需要向其他app釋出訊息。本學習將給出一個通過websocket來實現不同app之間訊息的訂購和釋出。 在小例子中,我們在所有節點之間都建立webSoc
iOS音訊格式之AMR和WAV互轉(更新支援64位)
---------------------------2016.6.15---------------------------------- ------- ---------------------------2015.8.25---------------------------------- --
block,inline和inline-block概念和區別(轉)
line eight wan pan 排列 isp 我們 .com 是個 總體概念 block和inline這兩個概念是簡略的說法,完整確切的說應該是 block-level elements (塊級元素) 和 inline elements (內聯元素)。block元素
python中xrange和range(轉)
log item .py 對象 nbsp net range all file 說到序列,我們第一想到的是一組有序元素組成的集合。同時,每個元素都有唯一的下標作為索引。 在Python中,有許多內界的序列。包括元組tuple,列表list,字符串str等。上面提到的序列
如何查找Linux中一些特殊數據類型定義,比如pid_t和uid_t(轉)
分享 文件 技術分享 定義 log 相關 uid linux源碼 pid 1. 查看man手冊,找到pid_t,可以通過getpid函數來看 2. 打開sys/types.h 3. 打開bits/types.h 4.
黃聰:濃縮的才是精華:淺析GIF格式圖片的存儲和壓縮(轉)
meid 單獨 圖片分辨率 change 之前 dex 本質 0.11 blog http://www.cnblogs.com/qcloud1001/p/6647080.html 成文迪, 在Web前端摸爬滾打的碼農一枚,對技術充滿熱情的菜鳥,致力為手Q的建設添磚加瓦
最小二乘法和最大似然估計的聯系和區別(轉)
enc bsp 聯系 角度 tro span nbsp sdn .science 對於最小二乘法,當從模型總體隨機抽取n組樣本觀測值後,最合理的參數估計量應該使得模型能最好地擬合樣本數據,也就是估計值和觀測值之差的平方和最小。而對於最大似然法,當從模型總體隨機抽取n組樣本觀
貝葉斯vs頻率派:武功到底哪家強?| 說人話的統計學·協和八(轉)
定義 這一 tps cbc 出發 上一條 習慣 做出 而已 回我們初次見識了統計學理論中的“獨孤九劍”——貝葉斯統計學(戳這裏回顧),它的起源便是大名鼎鼎的貝葉斯定理。 整個貝葉斯統計學的精髓可以用貝葉斯定理這一條式子來概括: 我們做數據分析,絕大多數情況下希望得到的是關於
jira安裝和配置(轉)
com 文章 講解 art setup 軟件安裝包 使用 基於 tran Jira 安裝和配置 Jira 安裝和配置 Jira 安裝 Jira 配置 資料 Jira 安裝 Jira 安裝 官網:https://www.atlassian.com/softw
jenkins安裝和配置(轉)
quest tac 軟件安裝包 setup and all pro 地址 新版 轉載來自:http://wiki.jikexueyuan.com/project/linux-in-eye-of-java/Jenkins-Install-And-Settings.html
集成學習算法總結----Boosting和Bagging(轉)
原理 過程 訓練 嚴重 oos 機器學習 ppr 次數 error 1、集成學習概述 1.1 集成學習概述 集成學習在機器學習算法中具有較高的準去率,不足之處就是模型的訓練過程可能比較復雜,效率不是很高。目前接觸較多的集成學習主要有2種:基於Boosting的和基於Bagg
C#中的new和override(轉)
方法 too 找到 ring 父類 類繼承 衍生 virtual func 在衍生類中的方法上使用new和override關鍵字有何意義,可以通過一系列問題來找到答案。先看一段代碼: 1 class Program 2 { 3 s
iOS多線程編程--NSOperation(轉)
https 同時 寫到 elf 調整 ber ios多線程 bool per 這篇文章寫得非常不錯,基礎用法都涉及到了,我把文章提到的例子都寫到了demo裏面, 原文地址: iOS多線程--徹底學會多線程之『NSOperation』 demo下載:https://githu
C#可擴展編程之MEF學習筆記(三):導出類的方法和屬性(轉)
學習 說了 如何 mod ati dem ont num imp 前面說完了導入和導出的幾種方法,如果大家細心的話會註意到前面我們導出的都是類,那麽方法和屬性能不能導出呢???答案是肯定的,下面就來說下MEF是如何導出方法和屬性的。 還是前面的代碼,第二篇中已經提供了下
IOS推送之PHP代碼(轉)
推送消息 獲取 option code bsp 包含 connect 手機 pan 準備工作 1.獲取手機註冊應用的deviceToken(iphone手機註冊應用時返回唯一值deviceToken) 2.獲取ck.pem文件(做手機端的給) 3.獲取pass phrase
在Spring Boot中使用Spring-data-jpa實現分頁查詢(轉)
分頁查詢 log def inpu database ext identity odin btn 在我們平時的工作中,查詢列表在我們的系統中基本隨處可見,那麽我們如何使用jpa進行多條件查詢以及查詢列表分頁呢?下面我將介紹兩種多條件查詢方式。 1、引入起步依賴
通俗易懂理解TCP和UDP(轉)
建立 休息 如果 str 否則 無連接 一分鐘 tis 得到 知乎看到一位大牛“車小胖”的類比很貼切,就轉過來了。 原文鏈接:https://www.zhihu.com/question/51388497 或者:https://daily.zhihu.com/story/9
C語言 二維數組(指針)動態分配和釋放(轉)
i++ 進制 numbers 很多 print 算術 uil 換算 som C 二維數組(指針)動態分配和釋放 先明確下概念: 所謂32位處理器就是一次只能處理32位,也就是4個字節的數據,而64位處理器一次就能處理64位,即8個字節的數據。如果我們將總長128位的指令分別
hashCode() 和equals() 區別和作用(轉)
person set集合 static out fin 解決 詳細 返回 art 出處:https://www.jianshu.com/p/5a7f5f786b75 本章的內容主要解決下面幾個問題: 1 equals() 的作用是什麽? 2 equal