音频压缩工具——Speex的使用
尊重原创,这篇文章是我转载的原文地址 http://blog.csdn.net/xyz_lmn/article/details/8013490
***********************************************************************
* 转载务必在明显处注明:http://blog.csdn.net/xyz_lmn
* 作者:张兴业
* 邮箱:xy-zhang@163.com
***********************************************************************
以前在应用中使用到了Speex编解码,近来总结了一下Speex在Android上的实现。Speex是一套主要针对语音的开源免费,无专利保护的音频压缩格式。Speex工程着力于通过提供一个可以替代高性能语音编解码来降低语音应用输入门槛 。另外,相对于其它编解码,Speex也很适合网络应用,在网络应用上有着自己独特的优势。同时,Speex还是GNU工程的一部分,在改版的BSD协议中得到了很好的支持。Speex是基于CELP并且专门为码率在2-44kbps的语音压缩而设计的。Speex源码是基于c语音实现的(也有Java实现,效率相对较低)。
1、去Speex官网下载最新Speex源码。
2、创建新的android工程,并创建jni文件夹。
3、把speex源码目录下的libspeex和include目录及其子目录文件全部拷贝到$project/jni目录下。
4、在jni目录下新增Android.mk文件,编辑内容如下:
- LOCAL_PATH := $(call my-dir)
- include $(CLEAR_VARS)
- LOCAL_MODULE:= libspeex
- LOCAL_CFLAGS = -DFIXED_POINT -DUSE_KISS_FFT -DEXPORT="" -UHAVE_CONFIG_H
- LOCAL_C_INCLUDES := $(LOCAL_PATH)/include
- LOCAL_SRC_FILES :=\
- libspeex/bits.c \
- libspeex/buffer.c \
- libspeex/cb_search.c \
- libspeex/exc_10_16_table.c \
- libspeex/exc_10_32_table.c \
- libspeex/exc_20_32_table.c \
- libspeex/exc_5_256_table.c \
- libspeex/exc_5_64_table.c \
- libspeex/exc_8_128_table.c \
- libspeex/fftwrap.c \
- libspeex/filterbank.c \
- libspeex/filters.c \
- libspeex/gain_table.c \
- libspeex/gain_table_lbr.c \
- libspeex/hexc_10_32_table.c \
- libspeex/hexc_table.c \
- libspeex/high_lsp_tables.c \
- libspeex/jitter.c \
- libspeex/kiss_fft.c \
- libspeex/kiss_fftr.c \
- libspeex/lpc.c \
- libspeex/lsp.c \
- libspeex/lsp_tables_nb.c \
- libspeex/ltp.c \
- libspeex/mdf.c \
- libspeex/modes.c \
- libspeex/modes_wb.c \
- libspeex/nb_celp.c \
- libspeex/preprocess.c \
- libspeex/quant_lsp.c \
- libspeex/resample.c \
- libspeex/sb_celp.c \
- libspeex/scal.c \
- libspeex/smallft.c \
- libspeex/speex.c \
- libspeex/speex_callbacks.c \
- libspeex/speex_header.c \
- libspeex/stereo.c \
- libspeex/vbr.c \
- libspeex/vq.c \
- libspeex/window.c \
- speex_jni.cpp \
- include $(BUILD_SHARED_LIBRARY)
LOCAL_PATH := $(call my-dir)include $(CLEAR_VARS)LOCAL_MODULE:= libspeex
LOCAL_CFLAGS = -DFIXED_POINT -DUSE_KISS_FFT -DEXPORT="" -UHAVE_CONFIG_H
LOCAL_C_INCLUDES := $(LOCAL_PATH)/includeLOCAL_SRC_FILES :=\
libspeex/bits.c \
libspeex/buffer.c \
libspeex/cb_search.c \
libspeex/exc_10_16_table.c \
libspeex/exc_10_32_table.c \
libspeex/exc_20_32_table.c \
libspeex/exc_5_256_table.c \
libspeex/exc_5_64_table.c \
libspeex/exc_8_128_table.c \
libspeex/fftwrap.c \
libspeex/filterbank.c \
libspeex/filters.c \
libspeex/gain_table.c \
libspeex/gain_table_lbr.c \
libspeex/hexc_10_32_table.c \
libspeex/hexc_table.c \
libspeex/high_lsp_tables.c \
libspeex/jitter.c \
libspeex/kiss_fft.c \
libspeex/kiss_fftr.c \
libspeex/lpc.c \
libspeex/lsp.c \
libspeex/lsp_tables_nb.c \
libspeex/ltp.c \
libspeex/mdf.c \
libspeex/modes.c \
libspeex/modes_wb.c \
libspeex/nb_celp.c \
libspeex/preprocess.c \
libspeex/quant_lsp.c \
libspeex/resample.c \
libspeex/sb_celp.c \
libspeex/scal.c \
libspeex/smallft.c \
libspeex/speex.c \
libspeex/speex_callbacks.c \
libspeex/speex_header.c \
libspeex/stereo.c \
libspeex/vbr.c \
libspeex/vq.c \
libspeex/window.c \
speex_jni.cpp \include $(BUILD_SHARED_LIBRARY)
5.在jni目录下新增Application.mk文件,编辑内容如下
- APP_ABI := armeabi armeabi-v7a
APP_ABI := armeabi armeabi-v7a
6.在$project/jni/include/speex/目录下新增speex_config_types.h文件,编辑内容如下
- 01.#ifndef __SPEEX_TYPES_H__
- 02.#define __SPEEX_TYPES_H__
- 03.typedef short spx_int16_t;
- 04.typedef unsigned short spx_uint16_t;
- 05.typedef int spx_int32_t;
- 06.typedef unsigned int spx_uint32_t;
- 07.#endif
01.#ifndef __SPEEX_TYPES_H__
02.#define __SPEEX_TYPES_H__
03.typedef short spx_int16_t;
04.typedef unsigned short spx_uint16_t;
05.typedef int spx_int32_t;
06.typedef unsigned int spx_uint32_t;
07.#endif
7.创建JNI包装类speex_jni.cpp,用来调用Speex中的C代码函数,编辑内容如下
- #include <jni.h>
- #include <string.h>
- #include <unistd.h>
- #include <speex/speex.h>
- static int codec_open = 0;
- static int dec_frame_size;
- static int enc_frame_size;
- static SpeexBits ebits, dbits;
- void *enc_state;
- void *dec_state;
- static JavaVM *gJavaVM;
- extern "C"
- JNIEXPORT jint JNICALL Java_com_trunkbow_speextest_Speex_open
- (JNIEnv *env, jobject obj, jint compression) {
- int tmp;
- if (codec_open++ != 0)
- return (jint)0;
- speex_bits_init(&ebits);
- speex_bits_init(&dbits);
- enc_state = speex_encoder_init(&speex_nb_mode);
- dec_state = speex_decoder_init(&speex_nb_mode);
- tmp = compression;
- speex_encoder_ctl(enc_state, SPEEX_SET_QUALITY, &tmp);
- speex_encoder_ctl(enc_state, SPEEX_GET_FRAME_SIZE, &enc_frame_size);
- speex_decoder_ctl(dec_state, SPEEX_GET_FRAME_SIZE, &dec_frame_size);
- return (jint)0;
- }
- extern "C"
- JNIEXPORT jint Java_com_trunkbow_speextest_Speex_encode
- (JNIEnv *env, jobject obj, jshortArray lin, jint offset, jbyteArray encoded, jint size) {
- jshort buffer[enc_frame_size];
- jbyte output_buffer[enc_frame_size];
- int nsamples = (size-1)/enc_frame_size + 1;
- int i, tot_bytes = 0;
- if (!codec_open)
- return 0;
- speex_bits_reset(&ebits);
- for (i = 0; i < nsamples; i++) {
- env->GetShortArrayRegion(lin, offset + i*enc_frame_size, enc_frame_size, buffer);
- speex_encode_int(enc_state, buffer, &ebits);
- }
- //env->GetShortArrayRegion(lin, offset, enc_frame_size, buffer);
- //speex_encode_int(enc_state, buffer, &ebits);
- tot_bytes = speex_bits_write(&ebits, (char *)output_buffer,
- enc_frame_size);
- env->SetByteArrayRegion(encoded, 0, tot_bytes,
- output_buffer);
- return (jint)tot_bytes;
- }
- extern "C"
- JNIEXPORT jint JNICALL Java_com_trunkbow_speextest_Speex_decode
- (JNIEnv *env, jobject obj, jbyteArray encoded, jshortArray lin, jint size) {
- jbyte buffer[dec_frame_size];
- jshort output_buffer[dec_frame_size];
- jsize encoded_length = size;
- if (!codec_open)
- return 0;
- env->GetByteArrayRegion(encoded, 0, encoded_length, buffer);
- speex_bits_read_from(&dbits, (char *)buffer, encoded_length);
- speex_decode_int(dec_state, &dbits, output_buffer);
- env->SetShortArrayRegion(lin, 0, dec_frame_size,
- output_buffer);
- return (jint)dec_frame_size;
- }
- extern "C"
- JNIEXPORT jint JNICALL Java_com_trunkbow_speextest_Speex_getFrameSize
- (JNIEnv *env, jobject obj) {
- if (!codec_open)
- return 0;
- return (jint)enc_frame_size;
- }
- extern "C"
- JNIEXPORT void JNICALL Java_com_trunkbow_speextest_Speex_close
- (JNIEnv *env, jobject obj) {
- if (--codec_open != 0)
- return;
- speex_bits_destroy(&ebits);
- speex_bits_destroy(&dbits);
- speex_decoder_destroy(dec_state);
- speex_encoder_destroy(enc_state);
- }
#include <jni.h>#include <string.h>
#include <unistd.h>#include <speex/speex.h>static int codec_open = 0;static int dec_frame_size;
static int enc_frame_size;static SpeexBits ebits, dbits;
void *enc_state;
void *dec_state;static JavaVM *gJavaVM;extern "C"
JNIEXPORT jint JNICALL Java_com_trunkbow_speextest_Speex_open(JNIEnv *env, jobject obj, jint compression) {int tmp;if (codec_open++ != 0)return (jint)0;speex_bits_init(&ebits);speex_bits_init(&dbits);enc_state = speex_encoder_init(&speex_nb_mode);dec_state = speex_decoder_init(&speex_nb_mode);tmp = compression;speex_encoder_ctl(enc_state, SPEEX_SET_QUALITY, &tmp);speex_encoder_ctl(enc_state, SPEEX_GET_FRAME_SIZE, &enc_frame_size);speex_decoder_ctl(dec_state, SPEEX_GET_FRAME_SIZE, &dec_frame_size);return (jint)0;
}extern "C"
JNIEXPORT jint Java_com_trunkbow_speextest_Speex_encode(JNIEnv *env, jobject obj, jshortArray lin, jint offset, jbyteArray encoded, jint size) {jshort buffer[enc_frame_size];jbyte output_buffer[enc_frame_size];int nsamples = (size-1)/enc_frame_size + 1;int i, tot_bytes = 0;if (!codec_open)return 0;speex_bits_reset(&ebits);for (i = 0; i < nsamples; i++) {env->GetShortArrayRegion(lin, offset + i*enc_frame_size, enc_frame_size, buffer);speex_encode_int(enc_state, buffer, &ebits);}//env->GetShortArrayRegion(lin, offset, enc_frame_size, buffer);//speex_encode_int(enc_state, buffer, &ebits);tot_bytes = speex_bits_write(&ebits, (char *)output_buffer,enc_frame_size);env->SetByteArrayRegion(encoded, 0, tot_bytes,output_buffer);return (jint)tot_bytes;
}extern "C"
JNIEXPORT jint JNICALL Java_com_trunkbow_speextest_Speex_decode(JNIEnv *env, jobject obj, jbyteArray encoded, jshortArray lin, jint size) {jbyte buffer[dec_frame_size];jshort output_buffer[dec_frame_size];jsize encoded_length = size;if (!codec_open)return 0;env->GetByteArrayRegion(encoded, 0, encoded_length, buffer);speex_bits_read_from(&dbits, (char *)buffer, encoded_length);speex_decode_int(dec_state, &dbits, output_buffer);env->SetShortArrayRegion(lin, 0, dec_frame_size,output_buffer);return (jint)dec_frame_size;
}extern "C"
JNIEXPORT jint JNICALL Java_com_trunkbow_speextest_Speex_getFrameSize(JNIEnv *env, jobject obj) {if (!codec_open)return 0;return (jint)enc_frame_size;}extern "C"
JNIEXPORT void JNICALL Java_com_trunkbow_speextest_Speex_close(JNIEnv *env, jobject obj) {if (--codec_open != 0)return;speex_bits_destroy(&ebits);speex_bits_destroy(&dbits);speex_decoder_destroy(dec_state);speex_encoder_destroy(enc_state);
}
8.在Java层创建Speex工具类,内容如下:
- package com.trunkbow.speextest;
- public class Speex {
- /* quality
- * 1 : 4kbps (very noticeable artifacts, usually intelligible)
- * 2 : 6kbps (very noticeable artifacts, good intelligibility)
- * 4 : 8kbps (noticeable artifacts sometimes)
- * 6 : 11kpbs (artifacts usually only noticeable with headphones)
- * 8 : 15kbps (artifacts not usually noticeable)
- */
- private static final int DEFAULT_COMPRESSION = 8;
- Speex() {
- }
- public void init() {
- load();
- open(DEFAULT_COMPRESSION);
- }
- private void load() {
- try {
- System.loadLibrary("speex");
- } catch (Throwable e) {
- e.printStackTrace();
- }
- }
- public native int open(int compression);
- public native int getFrameSize();
- public native int decode(byte encoded[], short lin[], int size);
- public native int encode(short lin[], int offset, byte encoded[], int size);
- public native void close();
- }
package com.trunkbow.speextest;public class Speex {/* quality * 1 : 4kbps (very noticeable artifacts, usually intelligible) * 2 : 6kbps (very noticeable artifacts, good intelligibility) * 4 : 8kbps (noticeable artifacts sometimes) * 6 : 11kpbs (artifacts usually only noticeable with headphones) * 8 : 15kbps (artifacts not usually noticeable) */ private static final int DEFAULT_COMPRESSION = 8; Speex() { } public void init() { load(); open(DEFAULT_COMPRESSION); } private void load() { try { System.loadLibrary("speex"); } catch (Throwable e) { e.printStackTrace(); } } public native int open(int compression); public native int getFrameSize(); public native int decode(byte encoded[], short lin[], int size); public native int encode(short lin[], int offset, byte encoded[], int size); public native void close();
}
9、使用cygwin编译,生成so文件。
参考:
http://code.google.com/p/android-recorder/downloads/list 这个是一个android recorder ,使用speex编码(这个例子我在转载时就不能用了 ),
音频压缩工具——Speex的使用相关推荐
- 音频压缩工具——Speex的使用(AndroidiOS语音录音技术预演)
要在Android/iOS端实现语音对讲,原型为微信与米聊,开始预演所用技术,找到以下资料. 在Android开发中,需要录音并发送到对方设备上.这时问题来了,手机常会是GPRS.3G等方式上网,所以 ...
- 音频压缩SPEEX应用,对讲机
音频压缩SPEEX应用,对讲机 转载于:https://www.cnblogs.com/LittleTiger/p/10979563.html
- 音频编码-speex库的使用方法
音频编码-speex库的使用方法 Speex是近年来开发出的一套功能强大的语音引擎,能够实现高质量和低比特率的编码.它不仅提供了基于码激励线性预测(CELP)算法的编/解码模块,而且在其最新发布的版本 ...
- php 调用speex解码库,音频算法speex中的aec分析以及解析
算法原理: Speex的AEC是以NLMS(Normalized Least Mean Square)为基础,用MDF(multidelay block frequency domain)频域实现,最 ...
- node.js 用typescript实现 图片,音频压缩工具
首先让美术爸爸,帮给导出两张图片,分别为jpg和png格式 我们再来看看压缩前后的图片和音频质量,大小 我们发现图片质量不变,内存却减少了很多 我们再从网上下载一首笔主喜欢听的歌曲来压缩一下试试 我们 ...
- 采集音频和摄像头视频并实时H264编码及AAC编码
0. 前言 我在前两篇文章中写了DirectShow捕获音视频然后生成avi,再进行264编码的方法.那种方法有一些局限性,不适合实时性质的应用,如:视频会议.视频聊天.视频监控等.本文所使用的技术, ...
- 采集音频和摄像头视频并实时H264编码及AAC编码[转]
0. 前言 我在前两篇文章中写了DirectShow捕获音视频然后生成avi,再进行264编码的方法.那种方法有一些局限性,不适合实时性质的应用,如:视频会议.视频聊天.视频监控等.本文所使用的技术, ...
- Speex 一个开源的声学回声消除器(Acoustic Echo Cancellation)
在一般的VOIP软件或视频会议系统中,假设我们只有A和B两个人在通话,首先,A的声音传给B,B然后用喇叭放出来,而这时B的MIC呢则会采集到喇叭放出来的声音,然后传回给A,如果这个传输的过程中时延足够 ...
- 音频编辑开发SDK Audio DJ Studio for .NET Crack
11.7版本--Audio DJ Studio for .NET是 MultiMedia Soft 开发的 .NET Windows Forms 自定义控件,可以轻松地向使用Microsoft Vis ...
最新文章
- centos查看特定程序占用端口情况
- Interview:算法岗位面试—11.15下午上海某航天***公司(国企)技术面之工业机器视觉认知、计算机视觉算法的理解、目标检测相关项目案例
- 3DSlicer9:FAQ-3
- AutoDim:自动Embedding维度寻优,如何节省70%的存储空间同时还能大幅提效?
- 计算机网络资料篇(二)——快速理解网络协议
- php js 复选框选中,为每个选中的复选框显示相同的一组问题。 (PHP和JS / Jquery)...
- 你知道考的是平方数吗(记洛谷P1876题RE+WA的经历,Java语言描述)
- 【Interfacenavigation】按钮(29)
- 使用 Design Pattern Toolkit 进行模型驱动的开发入门
- 王道 计算机网络试题讲解_王道考研 计算机网络(7)学习笔记
- matlab信号探测,信号检测与估matlab仿真报告.doc
- mtk开机声音与关机声音修改(原创)
- 关于STC8H8K64U单片机IAP升级过程
- 基于HTML+CSS+JavaScript的在线图书阅读网页设计
- 如何零成本实现微信公众号自助查券返利机器人(六)
- 金弘同创:拼多多怎么退保证金
- 微信如何群发不封号?
- Pandas数据可视化的备忘录
- linux查看进程并kill
- 3.高精度加法(使用C++的string实现高精度加法运算的实例代码)包含auto解决办法