opus 压缩率_iOS 下Opus 压缩PCM音频数据方法

本文介绍如何在iOS中使用Opus库进行PCM音频数据的编码和解码。通过创建OpusEncoder和OpusDecoder对象,设置编码参数,并提供编码、解码方法,实现了音频数据的压缩和还原。代码示例展示了从音频数据到Opus编码数据的转换,以及从Opus数据回溯到原始音频数据的过程。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

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

不多说直接贴代码了

opusCodec.h

#import

@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

使用方法

首先 使用audioqueue 来录制音频数据 生成NSdata 类型的数据流

注意:使用audioqueue录制音频的大小最好设置为320

然后使用我上面的方法 opusInit 初始化 opus 编码器

使用-(NSData)encodePCMData:(NSData)data;

编码数据流

然后通过socket 上传服务器 通过服务器分发到手机上

手机收到服务器分发的数据使用

-(NSData)decodeOpusData:(NSData)data;

解码数据流

得到 可以播放的音频数据流

然后使用audio queue 来播放

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

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值