【第22期】观点:IT 行业加班,到底有没有价值?

G711编解码(G711与PCM类型互转)【转】

转载 2016年08月30日 10:34:23

来自:http://blog.csdn.net/qq_24551315/article/details/51134689

总结网上的G711编解码, 将其一直到andorid系统上去, 通过jni的来进行G711和PCM 的互转,  由于对于这两种格式没有深入的了解,, 所以如果想深入的了解他们可以百度, 在这里只是实现这样的功能罢了.

g711.c 包含编码和解码的方法

[html] view plain copy
 在CODE上查看代码片派生到我的代码片
  1. //http://www.easydarwin.org/article/Streaming/38.html               参考文章  
  2. //https://github.com/EasyDarwin/EasyAACEncoder/blob/master/g711.cpp  
  3. //http://www.oschina.net/code/snippet_1173523_38946  
  4. //http://blog.csdn.net/rightorwrong/article/details/4209467  PCM 2 G711  G711 2 PCM   
  5.   
  6. unsigned char encode(short pcm);  
  7. short decode(unsigned char alaw);  
  8.   
  9. /** 个人理解  
  10.  * bitsize 应该为16, pBuffer(pcm数据) 两个char 合成一个 short ,长度自然就是原来的一半(nBufferSize/2),  
  11.  * 通过编码后short类型的数据变为char类型,让后复制给pCodecBits  
  12.  */  
  13. int g711_encode(unsigned char* pCodecBits, const char* pBuffer, int nBufferSize)  
  14. {  
  15.     short* buffer = (short*)pBuffer;  
  16.     int i;  
  17.     for(i=0; i<nBufferSize/2; i++)  
  18.     {  
  19.         pCodecBits[i] = encode(buffer[i]);  
  20.     }  
  21.   
  22.     return nBufferSize/2;  
  23. }  
  24.   
  25. /** 个人理解  
  26.  * bitsize 应该为16, pcm 数组的宽度变为原来两倍(short *out_data = (short*)pRawData;),  
  27.  * 通过对pBuffer(g711数据)中char解码转为两个字节的short ,后复制给out_data数组, 在使用的时候又转为char类型, 则 解码后的数据就是原来的两倍(nBufferSize*2)  
  28.  */  
  29. int g711_decode(char* pRawData, const unsigned char* pBuffer, int nBufferSize)  
  30. {  
  31.     short *out_data = (short*)pRawData;  
  32.     int i;  
  33.     for(i=0; i<nBufferSize; i++)  
  34.     {  
  35.         out_data[i] = decode(pBuffer[i]);  
  36.     }  
  37.   
  38.     return nBufferSize*2;  
  39. }  
  40.   
  41. #define MAX 32635  
  42. unsigned char encode(short pcm)  
  43. {  
  44.     int sign = (pcm & 0x8000) >> 8;  
  45.     if (sign != 0)  
  46.         pcm = -pcm;  
  47.     if (pcm > MAX) pcm = MAX;  
  48.     int exponent = 7;  
  49.     int expMask;  
  50.     for (expMask = 0x4000; (pcm & expMask) == 0  
  51.         && exponent>0; exponent--, expMask >>= 1) { }  
  52.     int mantissa = (pcm >> ((exponent == 0) ? 4 : (exponent + 3))) & 0x0f;  
  53.     unsigned char alaw = (unsigned char)(sign | exponent << 4 | mantissa);  
  54.     return (unsigned char)(alaw^0xD5);  
  55. }  
  56.   
  57. short decode(unsigned char alaw)  
  58. {  
  59.     alaw ^= 0xD5;  
  60.     int sign = alaw & 0x80;  
  61.     int exponent = (alaw & 0x70) >> 4;  
  62.     int data = alaw & 0x0f;  
  63.     data <<= 4;  
  64.     data += 8;  
  65.     if (exponent != 0)  
  66.         data += 0x100;  
  67.     if (exponent > 1)  
  68.         data <<= (exponent - 1);  
  69.   
  70.     return (short)(sign == 0 ? data : -data);  
  71. }  

jni中实现编解码,只贴出主要代码

G711 -> PCM

[html] view plain copy
 在CODE上查看代码片派生到我的代码片
  1. JNIEXPORT void JNICALL Java_com_ff_aacdemo_jni_G711Coder_g711Topcm  
  2.   (JNIEnv *env, jobject obj){  
  3.     FILE* fpOut = fopen("/storage/emulated/0/t/pcm_to_g711.pcm", "wb");  
  4.     FILE* fpIn = fopen("/storage/emulated/0/t/pcm_to_g711.g711", "rb");//g711格式文件在最后会给出连接  
  5.   
  6.     int g711_BufferSize = 1024;  
  7.     char g711_Buffer[g711_BufferSize];  
  8.   
  9.     int len;  
  10.     while((len = fread(g711_Buffer, 1, g711_BufferSize, fpIn)) > 0){  
  11.         LOGD("g711topcm length = %d", len);  
  12.         char pcmBuffer[len];  
  13.         int pcmbufsize = g711_decode(pcmBuffer, g711_Buffer, len);  
  14.         fwrite(pcmBuffer, 1, pcmbufsize, fpOut);  
  15.     }  
  16.     LOGD("g711topcm end");  
  17.     fclose(fpIn);  
  18.     fclose(fpOut);  
  19. }  

PCM -> G711

[html] view plain copy
 在CODE上查看代码片派生到我的代码片
  1. JNIEXPORT void JNICALL Java_com_ff_aacdemo_jni_G711Coder_pcmTog711  
  2.   (JNIEnv *env, jobject obj){  
  3.     FILE* fpIn = fopen("/storage/emulated/0/t/pcm_to_g711.pcm", "rb");  
  4.     FILE* fpOut = fopen("/storage/emulated/0/t/pcm_to_g711.g711", "wb");  
  5.   
  6.     LOGD("pcmTog711 1");  
  7.   
  8.     int pcm_BufferSize = 1024;  
  9.     char pcm_Buffer[pcm_BufferSize];  
  10.   
  11.     int len;  
  12.     while((len = fread(pcm_Buffer, 1, pcm_BufferSize, fpIn)) > 0){  
  13.         LOGD("pcmTog711 length = %d", len);  
  14.         char g711Buffer[len];  
  15.         int  g711BufSize = g711_encode(g711Buffer, pcm_Buffer, len);  
  16.         fwrite(g711Buffer, 1, g711BufSize, fpOut);  
  17.     }  
  18.     LOGD("pcmTog711 end");  
  19.     fclose(fpIn);  
  20.     fclose(fpOut);  
  21. }  

虽然能够实现他们的相互转换, 可是,不怎么好找到播放器播放, 效果始终没有验证, 在下一章中, 将总结将 G711 转换为 AAC , AAC这种格式将能够使用常规的播放器进行播放

g711文件 http://download.csdn.net/detail/yuanzhenhai/2693878

举报

相关文章推荐

G711编解码(G711与PCM类型互转)

总结网上的G711编解码, 将其一直到andorid系统上去, 通过jni的来进行G711和PCM 的互转,  由于对于这两种格式没有深入的了解,, 所以如果想深入的了解他们可以百度, 在这里只是实现...

FFMPEG实现的转码程序

本例子是由FFEMPG的doc/example例子transcode.c修改而来,可以根据需求任意转换音视频的编码。 原来的例子的作用更类似于remux,并没有实现转码的功能,只是实现了格式转换,...

程序员升职加薪指南!还缺一个“证”!

CSDN出品,立即查看!

AndroidJNI入门之G.711音频编解码

需求背景:博主目前所在的公司是一家做视频通讯的公司,所以对音频,视频这一块对编码方式都有一定的要求,由于之前一直没有接触JNI这一块,突然让我去做音频的转码还是有一定的苦难的。一开始对于JNI编程我是...

Android 录音PCM 转G711U,非常简单,非FFMPEG

最近在开发对讲机时,由于G711数据压缩的比较好,适合网络传输,所以音频采用 了 G711编码。但是Android里面的AudioRecord编码出来的是 PCM数据,所以要对数据进行编码后再发送。...

Android G711编解码

G.711 Android 编解码库源码

Android/Java 实现PCM与G.711编码互转

背景最近做一个UDP局域网通讯的项目,是 Android 手机和一个室内硬件外设通讯,因为这个外设的音频传输采用的是 G.711 a率 编码,Android 的 AudioRecord 又不能直接采集...

G711编解码

搞语音对讲几天了,播放时声音干扰太大了。拖得时间久有两个原因:1.每次采样的位数这个值设置的问题。本来是用的采样位数为16,但是服务端那边说用8。导致编解码时8位的始终有杂音。1.G711编解码的问题...

用于音频对讲: 解g711u数据流和压缩pcm的数据流的方法

//g711u2pcm int UlawCodec_decode(int16_t *samples, int count, void *payload, int length) { int8_t...

ADPCM音频格式 编解码

ADPCM音频格式 编解码

G711格式音频编解码

G711 音频解码
收藏助手
不良信息举报
您举报文章:深度学习:神经网络中的前向传播和反向传播算法推导
举报原因:
原因补充:

(最多只允许输入30个字)