美文网首页iOSAVFoundation 拍片专用
iOS 下Opus 压缩PCM音频数据方法

iOS 下Opus 压缩PCM音频数据方法

作者: shuolol | 来源:发表于2016-10-10 19:06 被阅读1132次

    我上篇文章有交大家如何编译Opus库

    不多说直接贴代码了

    
    opusCodec.h
    
    #import <Foundation/Foundation.h>
    @interface opusCodec : NSObject
    
    -(void)opusInit;
    
    -(NSData*)encodePCMData:(NSData*)data;
    
    -(NSData*)decodeOpusData:(NSData*)data;
    
    -(void)destroy;
    
    @end
    
    
    
    opusCodec.m
    
    #import "opusCodec.h"
    
    #import "opus.h"
    
    #define kDefaultSampleRate 8000
    
    #define WB_FRAME_SIZE  320
    
    @implementation opusCodec
    
    {
    
    OpusEncoder *enc;
    
    OpusDecoder *dec;
    
    unsigned char opus_data_encoder[40];
    
    }
    
    -(void)opusInit
    
    {
    
    int error;
    
    enc = opus_encoder_create(kDefaultSampleRate, 1, OPUS_APPLICATION_VOIP, &error);//(采样率,声道数,,)
    
    dec = opus_decoder_create(kDefaultSampleRate, 1, &error);
    
    opus_encoder_ctl(enc, OPUS_SET_BITRATE(kDefaultSampleRate));//比特率
    
    opus_encoder_ctl(enc, OPUS_SET_BANDWIDTH(OPUS_AUTO));//OPUS_BANDWIDTH_NARROWBAND 宽带窄带
    
    opus_encoder_ctl(enc, OPUS_SET_VBR(0));
    
    opus_encoder_ctl(enc, OPUS_SET_VBR_CONSTRAINT(1));
    
    opus_encoder_ctl(enc, OPUS_SET_COMPLEXITY(8));//录制质量 1-10
    
    opus_encoder_ctl(enc, OPUS_SET_PACKET_LOSS_PERC(0));
    
    opus_encoder_ctl(enc, OPUS_SET_SIGNAL(OPUS_SIGNAL_VOICE));//信号
    
    }
    
    - (NSData *)encode:(short *)pcmBuffer length:(NSInteger)lengthOfShorts
    
    {
    
    //    NSLog(@"--->>lengthOfShorts = %ld  size -= %lu",(long)lengthOfShorts,sizeof(short));
    
    int frame_size = (int)lengthOfShorts / sizeof(short);//WB_FRAME_SIZE;
    
    short input_frame[frame_size];
    
    opus_int32 max_data_bytes = 2 * WB_FRAME_SIZE ;//随便设大,此时为原始PCM大小
    
    memcpy(input_frame, pcmBuffer, lengthOfShorts );//frame_size * sizeof(short)
    
    int encodeBack = opus_encode(enc, input_frame, frame_size, opus_data_encoder, max_data_bytes);
    
    //    NSLog(@"encodeBack===%d",encodeBack);
    
    if (encodeBack > 0)
    
    {
    
    NSData *decodedData = [NSData dataWithBytes:opus_data_encoder length:encodeBack];
    
    return decodedData;
    
    }
    
    else
    
    {
    
    return nil;
    
    }
    
    }
    
    //int decode(unsigned char* in_data, int len, short* out_data, int* out_len) {
    
    -(int)decode:(unsigned char *)encodedBytes length:(int)lengthOfBytes output:(short *)decoded
    
    {
    
    int frame_size = WB_FRAME_SIZE;
    
    unsigned char cbits[frame_size*2];
    
    memcpy(cbits, encodedBytes, lengthOfBytes);
    
    int pcm_num = opus_decode(dec, cbits, lengthOfBytes, decoded, frame_size, 0);
    
    //    NSLog(@"解压后长度=%d",pcm_num);
    
    return frame_size;
    
    }
    
    -(NSData *)encodePCMData:(NSData*)data
    
    {
    
    //    NSLog(@"原始数据长度--->>%lu",(unsigned long)data.length);
    
    return  [self encode:(short *)[data bytes] length:[data length]];
    
    }
    
    -(NSData *)decodeOpusData:(NSData*)data
    
    {
    
    int len = (int)[data length];
    
    Byte *byteData = (Byte*)malloc(len);
    
    memcpy(byteData, [data bytes], len);
    
    int frame_size = WB_FRAME_SIZE;
    
    short decodedBuffer[frame_size ];
    
    int nDecodedByte = sizeof(short) * [self decode:byteData length:len output:decodedBuffer];
    
    NSData *PCMData = [NSData dataWithBytes:(Byte *)decodedBuffer length:nDecodedByte];
    
    return PCMData;
    
    }
    
    -(void)destroy
    
    {
    
    opus_encoder_destroy(enc);
    
    opus_decoder_destroy(dec);
    
    }
    
    @end
    
    

    使用方法

    1. 首先 使用audioqueue 来录制音频数据 生成NSdata 类型的数据流
      注意:使用audioqueue录制音频的大小最好设置为320
      然后使用我上面的方法 opusInit 初始化 opus 编码器

    2. 使用-(NSData)encodePCMData:(NSData)data;
      编码数据流
      然后通过socket 上传服务器 通过服务器分发到手机上

    3. 手机收到服务器分发的数据使用
      -(NSData)decodeOpusData:(NSData)data;
      解码数据流
      得到 可以播放的音频数据流
      然后使用audio queue 来播放

    4. 使用完以后记得使用-(void)destroy 来销毁

    相关文章

      网友评论

      • 烟花灬肆意:audioqueue 我设置的每个缓存是320,为什么有时候240,就调用了录音函数,正常应该320,这样导致压缩出现bug。
      • 烟花灬肆意:为什么内部解压pcm_num是320,返回的NSdata的数据是640
      • 来闹的:你好 我按照你的方法 audioqueue录制 opus压缩流 然后opus解压流用audioqueue播放
        为什么会有很大的杂音(类似快速的翻书声)
        来闹的:@shuolol我还有个关于audioqueue的问题想请教一下 用audioqueue实现边录边放 它自动停止了是什么鬼 和我设置的缓冲器有关系嚒
        来闹的:@shuolol 懂了 找到问题了 我录音流640了 谢谢
        shuolol:你 每个音频流大小 没有设置正确
      • b57bc247097e:这个会报错stepping may behave oddly; variables may not be available 难道你不会吗
      • b0d5dcd66709:楼主,有demo吗?传过来的不是(short *)buffer,用的AudioQueueBufferRef,总是运行出错
        b57bc247097e:老哥 现在有demo了吗
      • Thebloodelves:太直接了……

      本文标题:iOS 下Opus 压缩PCM音频数据方法

      本文链接:https://www.haomeiwen.com/subject/fmdvyttx.html