Opus从入门到精通(四)Opus解码程序实现

简介: 由于我们解码后的数据直接写入文件,无法通过vlc等播放器播放,我们通过lame将解码后的数据再编码成mp3,当然,你要乐意也可以直接价格wav头输出成wav. 通过采样率,声道数等创建lame编码器

前面我们已经成功将PCM文件编码成Opus,写入文件,接下来我们实现解码.


初始化解码器


解码器创建依赖于编码前数据的采样率,声道数:


OpusDecoder *pOpusDec = opus_decoder_create(sampleRateInHz, channelConfig, &error);
if (error != OPUS_OK)
{
    fprintf(stderr, "Cannot create decoder: %s\n", opus_strerror(error));
    goto failure;
}


创建MP3编码器


由于我们解码后的数据直接写入文件,无法通过vlc等播放器播放,我们通过lame将解码后的数据再编码成mp3,当然,你要乐意也可以直接价格wav头输出成wav. 通过采样率,声道数等创建lame编码器:


lame_global_flags *lame = NULL;
lame = lame_init();
lame_set_in_samplerate(lame, sampleRateInHz);//设置输入采样率
lame_set_num_channels(lame, channelConfig);//输入流的声道
lame_set_out_samplerate(lame, sampleRateInHz);//设置输出采样率
lame_set_brate(lame, 16);//设置码率
lame_set_quality(lame, 7);
lame_init_params(lame);


读取文件


前面我们编码的音频设置的码率是16kbits,并且一帧20毫秒,那么一帧的文件大小是16000 * 8  / 1000 * 20 = 40byte,我们每次读取一帧,循环读取文件:


#define READ_BUFFER_SIZE 40
#define WRITE_BUFFER_SIZE 520
fbytes = (unsigned char*)malloc(WRITE_BUFFER_SIZE);
out = (short *)malloc(640 * sizeof(short));
while (1)
{
    unsigned char data[READ_BUFFER_SIZE];
    num_read = fread(data, 1, READ_BUFFER_SIZE, fin);
    if (num_read > 0)
    {
        int output_samples = opus_decode(pOpusDec, data, num_read, out, DECODE_OUTPUT_SIZE, 0);
        int result = lame_encode_buffer(lame, out, out,output_samples, fbytes, WRITE_BUFFER_SIZE);
        if (fwrite(fbytes, 1, result, fout) != (unsigned)(result)){
            printf("write error,output_samples = %d, result = %d\n",output_samples,result);
            goto failure;
        }
    }
    else{
        int result = lame_encode_flush(lame, fbytes, WRITE_BUFFER_SIZE);
        if(result >= 0){
            if (fwrite(fbytes, 1, result, fout) != (unsigned)(result)){
                fprintf(stderr, "222 Error writing.\n");
                goto failure;
            }
        }
        lame_close(lame);
        break;
    }
}


存在问题


这样一个简单的opus解码器程序就出来了,但是这样有个问题,我们解码的时候必须事先知道原始音频的采样率,声道数,帧大小等. 而且必须保证编码后的数据每一帧大小都相同,这样就要求编码时必须配置成CBR,即固定码率. 我们假设有这么一个文件结构,里面不仅可以保持编码后的opus数据,而且可以保存帧率等信息,并且每帧之间都有帧间隔标记,这样就可以解决上面这些问题. 那么有没有这么一种文件结构呢?还真有,opus编码的标准封装ogg就是干这个事情的,后面的文章我们分析opus的ogg封装格式.


例代码地址:github.com/qingkouwei/…


如果对你有帮助的话点个赞吧!!!

目录
相关文章
|
编解码 API 语音技术
Opus从入门到精通(七)Opus编码基础之认识声音
前面我们分析完Opus的编解码api使用,封装原理等,接下来我们准备分析Opus编码原理.Opus编码是一个复杂的工作,我们需要做一些基本铺垫,包括认识声音,压缩编码基础.认识音频有助于我们了解音频特征,不仅对语音有助于我们理解编码技术,同时在语音识别,TTS等场景提供帮助
582 0
Opus从入门到精通(七)Opus编码基础之认识声音
|
存储 编解码 安全
Opus从入门到精通(二):编解码器使用
opus_encoder_get_size()返回编码器状态要求的大小。注意,这段代码的未来版本可能改变大小,所以没有assuptions应该对它做出。编码器状态在内存中总是连续,复制它只要一个浅拷贝就足够了。使用opus_encoder_ctl()接口可以改变一些编码器的参数设置。所有这些参数都已有缺省值,所以只在必要的情况下改变它们。
1397 0
|
2月前
|
编解码 语音技术 内存技术
FFmpeg开发笔记(五十八)把32位采样的MP3转换为16位的PCM音频
《FFmpeg开发实战:从零基础到短视频上线》一书中的“5.1.2 把音频流保存为PCM文件”章节介绍了将媒体文件中的音频流转换为原始PCM音频的方法。示例代码直接保存解码后的PCM数据,保留了原始音频的采样频率、声道数量和采样位数。但在实际应用中,有时需要特定规格的PCM音频。例如,某些语音识别引擎仅接受16位PCM数据,而标准MP3音频通常采用32位采样,因此需将32位MP3音频转换为16位PCM音频。
88 0
FFmpeg开发笔记(五十八)把32位采样的MP3转换为16位的PCM音频
|
1月前
|
编解码 监控 网络协议
如何使用FFmpeg实现RTSP推送H.264和H.265(HEVC)编码视频
本文详细介绍了如何使用FFmpeg实现RTSP推送H.264和H.265(HEVC)编码视频。内容涵盖环境搭建、编码配置、服务器端与客户端实现等方面,适合视频监控系统和直播平台等应用场景。通过具体命令和示例代码,帮助读者快速上手并实现目标。
277 6
|
7月前
|
安全 数据处理 数据格式
深入浅出:FFmpeg 音频解码与处理AVFrame全解析(三)
深入浅出:FFmpeg 音频解码与处理AVFrame全解析
335 0
|
7月前
|
存储 编解码 算法
深入浅出:FFmpeg 音频解码与处理AVFrame全解析(一)
深入浅出:FFmpeg 音频解码与处理AVFrame全解析
1421 0
|
计算机视觉
ffmpeg配置opus
ffmpeg配置opus
302 0
|
机器学习/深度学习 存储 编解码
Opus从入门到精通(八)Opus编码基础之压缩编码
莫尔斯码就是大家熟悉的电报码,它的发明为人类做出了巨大的贡献.该码采用"."和"-"来表示26个英文字母,这实质上还是二进制码(点为"0",而杠为"1"),但是它没有采用固定字长的编码方式,而是采用了常用字母用短码表示(如E用"."表示,T用"-"表示),不常用字母用长码表示(如Z用"--.."表示,j用"-..-"表示)的变长编码方式.通过对英文单词进行大量统计,找出各字母的概率,最后确定有12个字母出现概率最低,用4bit数字表示,有8个字母出现概率较低,用3bit数字表示;有4个字母出现概率较高,用2bit数字表示;有两个字母出现概率最高,用1bit表示,共26个字母.
630 0
|
存储 编解码 API
opus
Opus是一款完全开放的功能广泛的音频编解码器。Opus在互联网上的交互式语音和音乐传输方面无可匹敌,但也适用于存储和流媒体应用程序。它由Internet工程任务组(IETF)标准化为RFC 6716 ,该标准融合了Skype的SILK编解码器和Xiph.Org的CELT编解码器中的技术。
242 0
|
编解码 算法 数据格式
iOS音视频开发 - 音频编码格式(pcm、wav、mp3、aac、ogg)
我们通常从音乐App(如:网易云音乐)听歌时,会看到一首歌需要的存储空间大概是10M左右,对于手机磁盘来说这是可以接受的。但在网络中实时在线传播的话,这个数据量可能就太大了,所以必须对其进行压缩编码。