Android中Pcm文件转换为Mp3

刚完成了一个pcm转成mp3的小工作,记录下自己解决这个问题的过程,以便以后可以参考。pcm转换mp3首选的就是lame这个开源框架,下载地址lame,下载完成后需要ndk编译lame。安卓ndk环境配置可以百度。下面记录下ndk编译lame的过程

首先创建一个目录mp3lame(目录名字随意),然后在目录下创建jni文件夹,将lame源码下的libmp3lame文件拷贝到jni目录下,在jni目录下创建Android.mk

LOCAL_PATH := $(call my-dir)

include $(CLEAR_VARS)

LOCAL_MODULE        := libmp3lame
LOCAL_CFLAGS := -DSTDC_HEADERS
LOCAL_SRC_FILES     := \
./libmp3lame/bitstream.c \
./libmp3lame/encoder.c \
./libmp3lame/fft.c \
./libmp3lame/gain_analysis.c \
./libmp3lame/id3tag.c \
./libmp3lame/lame.c \
./libmp3lame/mpglib_interface.c \
./libmp3lame/newmdct.c \
./libmp3lame/presets.c \
./libmp3lame/psymodel.c \
./libmp3lame/quantize.c \
./libmp3lame/quantize_pvt.c \
./libmp3lame/reservoir.c \
./libmp3lame/set_get.c \
./libmp3lame/tables.c \
./libmp3lame/takehiro.c \
./libmp3lame/util.c \
./libmp3lame/vbrquantize.c \
./libmp3lame/VbrTag.c \
./libmp3lame/version.c \
./wrapper.c

LOCAL_LDLIBS := -llog

include $(BUILD_SHARED_LIBRARY)

在创建Application.mk

APP_PLATFORM := android-19

这是指定app编译的sdk的版本,不设置会在编译过程报错,还有其他的一些配置参数,可以百度下。

把这两个文件放到jni目录下面,在mp3lame目录下执行ndk-build命令,文件开始编译

Android中Pcm文件转换为Mp3_第1张图片

编写wrapper.c文件,这个是jni文件,需要自己编写方法去调用lame的方法

#include 
#include 
#include 
#include  
#include "libmp3lame/lame.h"

#define LOG_TAG "LAME ENCODER"
#define LOGD(format, args...)  __android_log_print(ANDROID_LOG_DEBUG, LOG_TAG, format, ##args);
#define BUFFER_SIZE 8192
#define be_short(s) ((short) ((unsigned short) (s) << 8) | ((unsigned short) (s) >> 8))

lame_t lame;

int read_samples(FILE *input_file, short *input) {
	int nb_read;
	nb_read = fread(input, 1, sizeof(short), input_file) / sizeof(short);

	int i = 0;
	while (i < nb_read) {
		input[i] = be_short(input[i]);
		i++;
	}

	return nb_read;
}

void Java_com_demo_iflytek_mscdemo_Lame_initEncoder(JNIEnv *env,
		jobject jobj, jint in_num_channels, jint in_samplerate, jint in_brate,
		jint in_mode, jint in_quality) {
	lame = lame_init();

	LOGD("Encoding Init parameters:");
	lame_set_num_channels(lame, in_num_channels);
	LOGD("Encoding Number of channels: %d", in_num_channels);
	lame_set_in_samplerate(lame, in_samplerate);
	LOGD("Encoding Sample rate: %d", in_samplerate);
	lame_set_brate(lame, in_brate);
	LOGD("Encoding Bitrate: %d", in_brate);
	lame_set_mode(lame, in_mode);
	LOGD("Encoding Mode: %d", in_mode);
	lame_set_quality(lame, in_quality);
	LOGD("Encoding Quality: %d", in_quality);

	int res = lame_init_params(lame);
	LOGD("Encoding Init returned: %d", res);
}

void Java_com_demo_iflytek_mscdemo_Lame_destroyEncoder(
		JNIEnv *env, jobject jobj) {
	int res = lame_close(lame);
	LOGD("Encoding Deinit returned: %d", res);
}

void Java_com_demo_iflytek_mscdemo_Lame_encodeFile(JNIEnv *env,
		jobject jobj, jstring in_source_path, jstring in_target_path) {
	const char *source_path, *target_path;
	source_path = (*env)->GetStringUTFChars(env, in_source_path, NULL);
	target_path = (*env)->GetStringUTFChars(env, in_target_path, NULL);

	FILE *input_file, *output_file;
	input_file = fopen(source_path, "rb");
	output_file = fopen(target_path, "wb");

	short input[BUFFER_SIZE];
	char output[BUFFER_SIZE];
	int nb_read = 0;
	int nb_write = 0;
	int nb_total = 0;

	LOGD("Encoding started");
	while (nb_read = read_samples(input_file, input)) {
		nb_write = lame_encode_buffer(lame, input, input, nb_read, output,
				BUFFER_SIZE);
		fwrite(output, nb_write, 1, output_file);
		nb_total += nb_write;
	}
	LOGD("Encoded %d bytes", nb_total);

	nb_write = lame_encode_flush(lame, output, BUFFER_SIZE);
	fwrite(output, nb_write, 1, output_file);
	LOGD("Encoded Flushed %d bytes", nb_write);

	fclose(input_file);
	fclose(output_file);
}

将文件放到jni同级目录下,重新编译一遍。

编写java类



public class Lame {
    static {
        System.loadLibrary("mp3lame");
    }

    /**
     *
     * @param numChannels 声道数
     * @param sampleRate 采样率
     * @param bitRate 比特率
     * @param mode 模式
     * @param quality
     */
    public native void initEncoder(int numChannels, int sampleRate, int bitRate, int mode, int quality);
    public native void destroyEncoder();
    public native int encodeFile(String sourcePath, String targetPath);
}

这样就可以调用方法了。

注意:直接转mp3会出现噪音。。因为安卓字节是小端排序,lame是大端排序,所以需要转换,转换代码如下:

/**
     * 大小端字节转换
     * @param fileName
     * @return
     * @throws IOException
     */
    public static String bigtolittle( String fileName) throws IOException {

        File file = new File(fileName);    //filename为pcm文件,请自行设置

        InputStream in = null;
        byte[] bytes = null;
        in = new FileInputStream(file);
        bytes = new byte[in.available()];//in.available()是得到文件的字节数
        int length = bytes.length;
        while (length != 1) {
            long i = in.read(bytes, 0, bytes.length);
            if (i == -1) {
                break;
            }
            length -= i;
        }

        int dataLength = bytes.length;
        int shortlength = dataLength / 2;
        ByteBuffer byteBuffer = ByteBuffer.wrap(bytes, 0, dataLength);
        ShortBuffer shortBuffer = byteBuffer.order(ByteOrder.LITTLE_ENDIAN).asShortBuffer();//此处设置大小端
        short[] shorts = new short[shortlength];
        shortBuffer.get(shorts, 0, shortlength);
        File file1 = File.createTempFile("pcm", null);//输出为临时文件
        String pcmtem = file1.getPath();
        FileOutputStream fos1 = new FileOutputStream(file1);
        BufferedOutputStream bos1 = new BufferedOutputStream(fos1);
        DataOutputStream dos1 = new DataOutputStream(bos1);
        for (int i = 0; i < shorts.length; i++) {
            dos1.writeShort(shorts[i]);

        }
        dos1.close();
        Log.d("gg", "bigtolittle: " + "=" + shorts.length);
        return pcmtem;
    }

ok这样就可以愉快的转换了,其他需求可以通过修改jni实现。放上几个资料的参考地址

https://blog.csdn.net/tcsupreme/article/details/80385670

https://www.jianshu.com/p/534741f5151c

你可能感兴趣的:(lame)