入门篇 - 2. AudioRecord 采集音频PCM并保存到文件

入门篇 - 1.ImageView、SurfaceView、自定义View 绘图
入门篇 - 2. AudioRecord 采集音频PCM并保存到文件

一、AudioRecord API详解

AudioRecord是Android系统提供的用于实现录音的功能类。

要想了解这个类的具体的说明和用法,我们可以去看一下官方的文档:

AndioRecord类的主要功能是让各种JAVA应用能够管理音频资源,以便它们通过此类能
够录制声音相关的硬件所收集的声音。
此功能的实现就是通过”pulling”(读取)AudioRecord对象的声音数据来完成的。
在录音过程中,应用所需要做的就是通过后面三个类方法中的一个去及时地获取AudioRecord对象的录音数据.
AudioRecord类提供的三个获取声音数据的方法分别是read(byte[], int, int), read(short[], int, int), read(ByteBuffer, int).
无论选择使用那一个方法都必须事先设定方便用户的声音数据的存储格式。

开始录音的时候,AudioRecord需要初始化一个相关联的声音buffer, 这个buffer主要是用来保存新的声音数据。这个buffer的大小,我们可以在对象构造期间去指定。
它表明一个AudioRecord对象还没有被读取(同步)声音数据前能录多长的音(即一次可以录制的声音容量)。
声音数据从音频硬件中被读出,数据大小不超过整个录音数据的大小(可以分多次读出),即每次读取初始化buffer容量的数据。

实现Android录音的流程为:

1. 构造一个AudioRecord对象,其中需要的最小录音缓存buffer大小可以通过getMinBufferSize方法得到。如果buffer容量过小,将导致对象构造的失败。
2. 初始化一个buffer,该buffer大于等于AudioRecord对象用于写声音数据的buffer大小。
3. 开始录音
4. 创建一个数据流,一边从AudioRecord中读取声音数据到初始化的buffer,一边将buffer中数据导入数据流。
5. 关闭数据流
6. 停止录音

二、使用 AudioRecord 实现录音,并生成wav

2.1 创建一个AudioRecord对象

首先要声明一些全局的变量参数:

privateAudioRecord audioRecord =null;// 声明 AudioRecord 对象
privateintrecordBufSize = 0;// 声明recoordBufffer的大小字段

获取buffer的大小并创建AudioRecord:

publicvoid createAudioRecord() {
     //audioRecord能接受的最小的buffer大小
     recordBufSize = AudioRecord.getMinBufferSize(frequency, channelConfiguration, EncodingBitRate);     

     audioRecord =new AudioRecord(MediaRecorder.AudioSource.MIC, frequency, channelConfiguration, EncodingBitRate, recordBufSize);
}

2.2 初始化一个buffer

bytedata[] =newbyte[recordBufSize];

2.3 开始录音

audioRecord.startRecording();
isRecording = true;

2.4 创建一个数据流,一边从AudioRecord中读取声音数据到初始化的buffer,一边将buffer中数据导入数据流。

FileOutputStream os =null;

try {
    os =new FileOutputStream(filename);
} catch (FileNotFoundException e) {
    e.printStackTrace();
}

if (null != os) {

    while (isRecording) {
        read = audioRecord.read(data, 0, recordBufSize);
        // 如果读取音频数据没有出现错误,就将数据写入到文件
        if (AudioRecord.ERROR_INVALID_OPERATION != read) {
            try {
                os.write(data);
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }

    try {
        os.close();
    } catch (IOException e) {
        e.printStackTrace();
    }
}

2.5 关闭数据流

修改标志位:isRecording 为false,上面的while循环就自动停止了,数据流也就停止流动了,Stream也就被关闭了。

isRecording =false;

2.6 停止录音

停止录音之后,注意要释放资源。

if(null!= audioRecord) {
    audioRecord.stop();
    audioRecord.release();
    audioRecord =null;
    recordingThread =null;
}

注:权限需求:WRITE_EXTERNAL_STORAGE、RECORD_AUDIO

到现在基本的录音的流程就介绍完了。但是这时候,有人就提出问题来了:

1)、我按照流程,把音频数据都输出到文件里面了,停止录音后,打开此文件,发现不能播放,到底是为什么呢?

答:按照流程走完了,数据是进去了,但是现在的文件里面的内容仅仅是最原始的音频数据,术语称为raw(中文解释是“原材料”或“未经处理的东西”),这时候,你让播放器去打开,它既不知道保存的格式是什么,又不知道如何进行解码操作。当然播放不了。

2)、那如何才能在播放器中播放我录制的内容呢?

答: 在文件的数据开头加入WAVE HEAD数据即可,也就是文件头。只有加上文件头部的数据,播放器才能正确的知道里面的内容到底是什么,进而能够正常的解析并播放里面的内容。具体的头文件的描述,在Play a WAV file on an AudioTrack里面可以进行了解。

添加WAVE文件头的代码如下:

publicclass PcmToWavUtil {
    privateint mBufferSize;    /* 缓存的音频大小*/
    privateint mSampleRate; /* 采样率*/
    privateint mChannel;        /* 声道数*/
    privateint mChannel;

    /**   
        * @param sampleRate sample rate、采样率
        * @param channel channel、声道
        * @param encoding Audio data format、音频格式
    */   
    PcmToWavUtil(intsampleRate,intchannel,int encoding) {
        this.mSampleRate = sampleRate;
        this.mChannel = channel;
        this.mBufferSize = AudioRecord.getMinBufferSize(mSampleRate, mChannel, encoding);
    }

    /**   
        * pcm文件转wav文件
        *
        * @param inFilename 源文件路径
        * @param outFilename 目标文件路径
    */
    publicvoid pcmToWav(String inFilename, String outFilename) {
        FileInputStream in;
        FileOutputStream out;
        long totalAudioLen;
        long totalDataLen;

        longlongSampleRate = mSampleRate;
        intchannels = mChannel == AudioFormat.CHANNEL_IN_MONO ? 1 : 2;
        longbyteRate = 16 * mSampleRate * channels / 8;
        byte[] data =newbyte[mBufferSize];

        try {
            in =new FileInputStream(inFilename);
            out =new FileOutputStream(outFilename);
            totalAudioLen = in.getChannel().size();
            totalDataLen = totalAudioLen + 36;
            writeWaveFileHeader(out, totalAudioLen, totalDataLen,
            longSampleRate, channels, byteRate);
            while(in.read(data) != -1) {
                out.write(data);
            }

            in.close();
            out.close();
        } catch (IOException e) {
            e.printStackTrace();
        }
     }


    /**    
        * 加入wav文件头
    */
    private void writeWaveFileHeader(FileOutputStream out,long totalAudioLen,longtotalDataLen,longlongSampleRate,intchannels,long byteRate) {
        throws IOException {
        byte[] header =newbyte[44];

        // RIFF/WAVE header
        header[0] = 'R';
        header[1] = 'I';
        header[2] = 'F';
        header[3] = 'F';
        header[4] = (byte) (totalDataLen & 0xff);
        header[5] = (byte) ((totalDataLen >> 8) & 0xff);
        header[6] = (byte) ((totalDataLen >> 16) & 0xff);
        header[7] = (byte) ((totalDataLen >> 24) & 0xff);
        
        //WAVE
        header[8] = 'W';
        header[9] = 'A';
        header[10] = 'V';
        header[11] = 'E';
        
        // 'fmt ' chunk
        header[12] = 'f';
        header[13] = 'm';
        header[14] = 't';
        header[15] = ' ';
        
        // 4 bytes: size of 'fmt ' chunk
        header[16] = 16;
        header[17] = 0;
        header[18] = 0;
        header[19] = 0;
  
        // format = 1
        header[20] = 1;
        header[21] = 0;
        header[22] = (byte) channels;
        header[23] = 0;
        header[24] = (byte) (longSampleRate & 0xff);
        header[25] = (byte) ((longSampleRate >> 8) & 0xff);
        header[26] = (byte) ((longSampleRate >> 16) & 0xff);
        header[27] = (byte) ((longSampleRate >> 24) & 0xff);
        header[28] = (byte) (byteRate & 0xff);
        header[29] = (byte) ((byteRate >> 8) & 0xff);
        header[30] = (byte) ((byteRate >> 16) & 0xff);
        header[31] = (byte) ((byteRate >> 24) & 0xff);
      
        // block align
        header[32] = (byte) (2 * 16 / 8);
        header[33] = 0;
        
        // bits per sample
        header[34] = 16;
        header[35] = 0;

        //data
        header[36] = 'd';
        header[37] = 'a';
        header[38] = 't';
        header[39] = 'a';
        header[40] = (byte) (totalAudioLen & 0xff);
        header[41] = (byte) ((totalAudioLen >> 8) & 0xff);
        header[42] = (byte) ((totalAudioLen >> 16) & 0xff);
        header[43] = (byte) ((totalAudioLen >> 24) & 0xff);

        out.write(header, 0, 44);
    }
}

三、附言

Android SDK 提供了两套音频采集的API,分别是:MediaRecorder 和 AudioRecord,前者是一个更加上层一点的API,它可以直接把手机麦克风录入的音频数据进行编码压缩(如AMR、MP3等)并存成文件,而后者则更接近底层,能够更加自由灵活地控制,可以得到原始的一帧帧PCM音频数据。如果想简单地做一个录音机,录制成音频文件,则推荐使用 MediaRecorder,而如果需要对音频做进一步的算法处理、或者采用第三方的编码库进行压缩、以及网络传输等应用,则建议使用 AudioRecord,其实 MediaRecorder 底层也是调用了 AudioRecord 与 Android Framework 层的 AudioFlinger 进行交互的。直播中实时采集音频自然是要用AudioRecord了。

四、源码

https://github.com/renhui/AudioDemo

你可能感兴趣的:(入门篇 - 2. AudioRecord 采集音频PCM并保存到文件)