Android利用AudioRecord类实现音频录制程序

AudioRecord类相对于MediaRecorder来说,更加接近底层,为我们封装的方法也更少。然而实现一个AudioRecord的音频录制程序也很简单。本实例代码如下:

package demo.camera;
import java.io.BufferedInputStream;
import java.io.BufferedOutputStream;
import java.io.DataInputStream;
import java.io.DataOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import android.app.Activity;
import android.content.ContentValues;
import android.content.Intent;
import android.hardware.Camera.AutoFocusCallback;
import android.media.AudioFormat;
import android.media.AudioManager;
import android.media.AudioRecord;
import android.media.AudioTrack;
import android.media.MediaPlayer;
import android.media.MediaRecorder;
import android.net.Uri;
import android.os.AsyncTask;
import android.os.Bundle;
import android.os.Environment;
import android.provider.MediaStore;
import android.util.Log;
import android.view.View;
import android.widget.Button;
import android.widget.TextView;
/**
 * 该实例中,我们使用AudioRecord类来完成我们的音频录制程序
 * AudioRecord类,我们可以使用三种不同的read方法来完成录制工作,
 * 每种方法都有其实用的场合
 * 一、实例化一个AudioRecord类我们需要传入几种参数
 * 1、AudioSource:这里可以是MediaRecorder.AudioSource.MIC
 * 2、SampleRateInHz:录制频率,可以为8000hz或者11025hz等,不同的硬件设备这个值不同
 * 3、ChannelConfig:录制通道,可以为AudioFormat.CHANNEL_CONFIGURATION_MONO和AudioFormat.CHANNEL_CONFIGURATION_STEREO
 * 4、AudioFormat:录制编码格式,可以为AudioFormat.ENCODING_16BIT和8BIT,其中16BIT的仿真性比8BIT好,但是需要消耗更多的电量和存储空间
 * 5、BufferSize:录制缓冲大小:可以通过getMinBufferSize来获取
 * 这样我们就可以实例化一个AudioRecord对象了
 * 二、创建一个文件,用于保存录制的内容
 * 同上篇
 * 三、打开一个输出流,指向创建的文件
 * DataOutputStream dos = new DataOutputStream(new BufferedOutputStream(new FileOutputStream(file)))
 * 四、现在就可以开始录制了,我们需要创建一个字节数组来存储从AudioRecorder中返回的音频数据,但是
 * 注意,我们定义的数组要小于定义AudioRecord时指定的那个BufferSize
 * short[]buffer = new short[BufferSize/4];
 * startRecording();
 * 然后一个循环,调用AudioRecord的read方法实现读取
 * 另外使用MediaPlayer是无法播放使用AudioRecord录制的音频的,为了实现播放,我们需要
 * 使用AudioTrack类来实现
 * AudioTrack类允许我们播放原始的音频数据
 *
 *
 * 一、实例化一个AudioTrack同样要传入几个参数
 * 1、StreamType:在AudioManager中有几个常量,其中一个是STREAM_MUSIC;
 * 2、SampleRateInHz:最好和AudioRecord使用的是同一个值
 * 3、ChannelConfig:同上
 * 4、AudioFormat:同上
 * 5、BufferSize:通过AudioTrack的静态方法getMinBufferSize来获取
 * 6、Mode:可以是AudioTrack.MODE_STREAM和MODE_STATIC,关于这两种不同之处,可以查阅文档
 * 二、打开一个输入流,指向刚刚录制内容保存的文件,然后开始播放,边读取边播放
 *
 * 实现时,音频的录制和播放分别使用两个AsyncTask来完成
 */
public class MyAudioRecord2 extends Activity{ 

  private TextView stateView; 

  private Button btnStart,btnStop,btnPlay,btnFinish; 

  private RecordTask recorder;
  private PlayTask player; 

  private File audioFile; 

  private boolean isRecording=true, isPlaying=false; //标记 

  private int frequence = 8000; //录制频率,单位hz.这里的值注意了,写的不好,可能实例化AudioRecord对象的时候,会出错。我开始写成11025就不行。这取决于硬件设备
  private int channelConfig = AudioFormat.CHANNEL_CONFIGURATION_MONO;
  private int audioEncoding = AudioFormat.ENCODING_PCM_16BIT; 

  public void onCreate(Bundle savedInstanceState){
    super.onCreate(savedInstanceState);
    setContentView(R.layout.my_audio_record); 

    stateView = (TextView)this.findViewById(R.id.view_state);
    stateView.setText("准备开始");
    btnStart = (Button)this.findViewById(R.id.btn_start);
    btnStop = (Button)this.findViewById(R.id.btn_stop);
    btnPlay = (Button)this.findViewById(R.id.btn_play);
    btnFinish = (Button)this.findViewById(R.id.btn_finish);
    btnFinish.setText("停止播放");
    btnStop.setEnabled(false);
    btnPlay.setEnabled(false);
    btnFinish.setEnabled(false); 

    //在这里我们创建一个文件,用于保存录制内容
    File fpath = new File(Environment.getExternalStorageDirectory().getAbsolutePath()+"/data/files/");
    fpath.mkdirs();//创建文件夹
    try {
      //创建临时文件,注意这里的格式为.pcm
      audioFile = File.createTempFile("recording", ".pcm", fpath);
    } catch (IOException e) {
      // TODO Auto-generated catch block
      e.printStackTrace();
    }
  } 

  public void onClick(View v){
    int id = v.getId();
    switch(id){
    case R.id.btn_start:
      //开始录制 

      //这里启动录制任务
      recorder = new RecordTask();
      recorder.execute(); 

      break;
    case R.id.btn_stop:
      //停止录制
      this.isRecording = false;
      //更新状态
      //在录制完成时设置,在RecordTask的onPostExecute中完成
      break;
    case R.id.btn_play: 

      player = new PlayTask();
      player.execute();
      break;
    case R.id.btn_finish:
      //完成播放
      this.isPlaying = false;
      break; 

    }
  } 

  class RecordTask extends AsyncTask<Void, Integer, Void>{
    @Override
    protected Void doInBackground(Void... arg0) {
      isRecording = true;
      try {
        //开通输出流到指定的文件
        DataOutputStream dos = new DataOutputStream(new BufferedOutputStream(new FileOutputStream(audioFile)));
        //根据定义好的几个配置,来获取合适的缓冲大小
        int bufferSize = AudioRecord.getMinBufferSize(frequence, channelConfig, audioEncoding);
        //实例化AudioRecord
        AudioRecord record = new AudioRecord(MediaRecorder.AudioSource.MIC, frequence, channelConfig, audioEncoding, bufferSize);
        //定义缓冲
        short[] buffer = new short[bufferSize]; 

        //开始录制
        record.startRecording(); 

        int r = 0; //存储录制进度
        //定义循环,根据isRecording的值来判断是否继续录制
        while(isRecording){
          //从bufferSize中读取字节,返回读取的short个数
          //这里老是出现buffer overflow,不知道是什么原因,试了好几个值,都没用,TODO:待解决
          int bufferReadResult = record.read(buffer, 0, buffer.length);
          //循环将buffer中的音频数据写入到OutputStream中
          for(int i=0; i<bufferReadResult; i++){
            dos.writeShort(buffer[i]);
          }
          publishProgress(new Integer(r)); //向UI线程报告当前进度
          r++; //自增进度值
        }
        //录制结束
        record.stop();
        Log.v("The DOS available:", "::"+audioFile.length());
        dos.close();
      } catch (Exception e) {
        // TODO: handle exception
      }
      return null;
    } 

    //当在上面方法中调用publishProgress时,该方法触发,该方法在UI线程中被执行
    protected void onProgressUpdate(Integer...progress){
      stateView.setText(progress[0].toString());
    } 

    protected void onPostExecute(Void result){
      btnStop.setEnabled(false);
      btnStart.setEnabled(true);
      btnPlay.setEnabled(true);
      btnFinish.setEnabled(false);
    } 

    protected void onPreExecute(){
      //stateView.setText("正在录制");
      btnStart.setEnabled(false);
      btnPlay.setEnabled(false);
      btnFinish.setEnabled(false);
      btnStop.setEnabled(true);
    } 

  } 

  class PlayTask extends AsyncTask<Void, Integer, Void>{
    @Override
    protected Void doInBackground(Void... arg0) {
      isPlaying = true;
      int bufferSize = AudioTrack.getMinBufferSize(frequence, channelConfig, audioEncoding);
      short[] buffer = new short[bufferSize/4];
      try {
        //定义输入流,将音频写入到AudioTrack类中,实现播放
        DataInputStream dis = new DataInputStream(new BufferedInputStream(new FileInputStream(audioFile)));
        //实例AudioTrack
        AudioTrack track = new AudioTrack(AudioManager.STREAM_MUSIC, frequence, channelConfig, audioEncoding, bufferSize, AudioTrack.MODE_STREAM);
        //开始播放
        track.play();
        //由于AudioTrack播放的是流,所以,我们需要一边播放一边读取
        while(isPlaying && dis.available()>0){
          int i = 0;
          while(dis.available()>0 && i<buffer.length){
            buffer[i] = dis.readShort();
            i++;
          }
          //然后将数据写入到AudioTrack中
          track.write(buffer, 0, buffer.length); 

        } 

        //播放结束
        track.stop();
        dis.close();
      } catch (Exception e) {
        // TODO: handle exception
      }
      return null;
    } 

    protected void onPostExecute(Void result){
      btnPlay.setEnabled(true);
      btnFinish.setEnabled(false);
      btnStart.setEnabled(true);
      btnStop.setEnabled(false);
    } 

    protected void onPreExecute(){  

      //stateView.setText("正在播放");
      btnStart.setEnabled(false);
      btnStop.setEnabled(false);
      btnPlay.setEnabled(false);
      btnFinish.setEnabled(true);
    } 

  }
}

可惜,本实例测试时有个问题,在录制的时候,会出现buffer over。缓存泄露,待解决。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

您可能感兴趣的文章:

  • Android音频录制MediaRecorder之简易的录音软件实现代码
  • Android使用AudioRecord实现暂停录音功能实例代码
  • Android录音--AudioRecord、MediaRecorder的使用
  • Android使用AudioRecord判断是否有音频输入
  • Android音频处理之通过AudioRecord去保存PCM文件进行录制,播放,停止,删除功能
  • Android提高之AudioRecord实现助听器的方法
  • android AudioRecorder简单心得分享
(0)

相关推荐

  • Android提高之AudioRecord实现助听器的方法

    通常来说,在进行Android项目开发的时候可以通过MediaRecorder和AudioRecord这两个工具来实现录音的功能,MediaRecorder直接把麦克风的数据存到文件,并且能够直接进行编码(如AMR,MP3等),而AudioRecord则是读取麦克风的音频流.本文使用AudioRecord读取音频流,使用AudioTrack播放音频流,通过"边读边播放"以及增大音量的方式来实现一个简单的助听器程序. 此处需要注意:由于目前的Android模拟器还不支持AudioReco

  • android AudioRecorder简单心得分享

    1.如何创建一个有效的AudioRecorder实例 Android各种设备的采样频率不同,输入的声道数也不同,如果采用固定的采样频率和声道数,那么得到的AudioRecorder不一定能够正常初始化.为了正常使用,需要尝试各种不同的参数,得到在此设备上可以用的AudioRecorder实例.代码如下: 复制代码 代码如下: private void createAudioRecord() {              for (int sampleRate : new int[]{44100,

  • Android使用AudioRecord实现暂停录音功能实例代码

    题外话:发现好久都没有上来写博文了,毕业设计加上公司暂时没有Android的项目做,只能去自学web上的知识,摸爬打滚到现在,花了一个多月时间根据公司的现有模板做了公司内部一个任务管理系统,感觉都是比较浅的知识,没什么可以写的.想到之前做的语音识别的项目,虽然现在没什么下文了,但是谁懂~~~将来呢? 言归正传,项目长这样子: 设计的思路: 由于自带的AudioRecord没有pauseRecord()方法,我把开始录音-->(暂停/继续录音)...-->停止录音叫做一次录音,点击一次暂停就会产

  • Android使用AudioRecord判断是否有音频输入

    Android党都应该玩过一个叫吹裙子的游戏,这个游戏就是原理就是通过监听用户吹出的气的力度来决定如何把MM的裙子弄飞起来的,所以关键在于如何判断用户吹气的力度问题.现在公司刚好有这个需求要评估就是需要一直监听用户的语音输入,当在两秒内没有语音输入时候就暂停某项事情,有语音输入的时候就要继续做某件事.其实这两件事情的本质原理是一样的,就是通过这AudioRecord来处理用户输入的原始音频数据,从而计算出当前用户输入的音量大小来判断用户是否有语音输入.下面贴上一段代码用户可以自行研究. publ

  • Android音频处理之通过AudioRecord去保存PCM文件进行录制,播放,停止,删除功能

    音频这方面很博大精深,我这里肯定讲不了什么高级的东西,最多也只是一些基础类知识,首先,我们要介绍一下Android他提供的录音类,实际上他有两个,一个是MediaRecorder,还有一个就是我们今天要用到的AudioRecord,那他们有什么区别呢? 一.区别 MediaRecorder和AudioRecord都可以录制音频,区别是MediaRecorder录制的音频文件是经过压缩后的,需要设置编码器.并且录制的音频文件可以用系统自带的Music播放器播放. 而AudioRecord录制的是P

  • Android录音--AudioRecord、MediaRecorder的使用

    Android提供了两个API用于实现录音功能:android.media.AudioRecord.android.media.MediaRecorder. 网上有很多谈论这两个类的资料.现在大致总结下: 1.AudioRecord 主要是实现边录边播(AudioRecord+AudioTrack)以及对音频的实时处理(如会说话的汤姆猫.语音) 优点:语音的实时处理,可以用代码实现各种音频的封装 缺点:输出是PCM语音数据,如果保存成音频文件,是不能够被播放器播放的,所以必须先写代码实现数据编码

  • Android音频录制MediaRecorder之简易的录音软件实现代码

    使用MediaRecorder的步骤:1.创建MediaRecorder对象2.调用MediRecorder对象的setAudioSource()方法设置声音的来源,一般传入MediaRecorder.MIC3.调用MediaRecorder对象的setOutputFormat()设置所录制的音频文件的格式4.调用MediaRecorder对象的setAudioRncoder().setAudioEncodingBitRate(int bitRate).setAudioSamlingRate(i

  • Android利用AudioRecord类实现音频录制程序

    AudioRecord类相对于MediaRecorder来说,更加接近底层,为我们封装的方法也更少.然而实现一个AudioRecord的音频录制程序也很简单.本实例代码如下: package demo.camera; import java.io.BufferedInputStream; import java.io.BufferedOutputStream; import java.io.DataInputStream; import java.io.DataOutputStream; imp

  • Android利用Canvas类绘制图形

    本文实例为大家分享了Android利用Canvas类绘制图形的具体代码,供大家参考,具体内容如下 首先介绍一下相关基础知识. 1.画笔(paint) 1.1.作用:画笔对象通过属性来设置笔的颜色.粗细.风格等. 1.2.设置图形绘制属性相关方法: setARGB():设置画笔的颜色:setAlpha():设置绘制图形的透明度:setStyle():设置画笔的样式风格:setStrockWidth():设置画笔的宽度: 1.3.设置文本绘制属性的相关方法: setTextSize():设置字体的大

  • Android使用MediaRecorder类进行录制视频

    我们提醒大家使用MediaRecorder录音录像的设置代码步骤一定要按照API指定的顺序来设置,否则报错 步骤为: 1.设置视频源,音频源,即输入源 2.设置输出格式 3.设置音视频的编码格式 一.首先看布局文件,这里有一个SurfaceView,这是一个绘制容器,可以直接从内存或者DMA等硬件接口取得图像数据, <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tool

  • 微信小程序音频录制波纹循环动画

    本文实例为大家分享了微信小程序音频录制波纹循环动画的具体代码,供大家参考,具体内容如下 实现的效果 第一种方法(利用微信小程序的animation) wxml部分 <!--pages/myRecode/myRecode.wxml--> <view class="myRecode">   <view class="recode" bindtouchstart='recodeClick' bindtouchend='recodeEnd'&g

  • Android开发之MediaPlayer多媒体(音频,视频)播放工具类

    本文实例讲述了Android开发之MediaPlayer多媒体(音频,视频)播放工具类.分享给大家供大家参考,具体如下: package com.android.imooc.chat; import java.io.IOException; import android.media.AudioManager; import android.media.MediaPlayer; import android.media.MediaPlayer.OnCompletionListener; impor

  • Android使用AudioRecord实现录音功能

    前言 Android使用AudioRecord实现录音 提示:以下是本篇文章正文内容,下面案例可供参考 一.AudioRecord使用 Android平台可以使用AudioRecord和MediaRecorder来实现录音,因为AudioRecord更接近底层,并且录制的数据为原始(pcm)数据,pcm数据可以再进行处理转换,直播中使用的都是处理后的pcm数据,所以在这里面学习下使用AudioRecord. 构造AudioRecord AudioRecord类的构造方法中有5个参数 /** *

  • Android编程实现的一键锁屏程序详解

    本文实例讲述了Android编程实现的一键锁屏程序.分享给大家供大家参考,具体如下: 据笔者了解,所有的Android手机都用电源键来手动锁屏.笔者使用手机时,一天之内手动锁屏的次数绝对不少于30次.如果都用电源键来进行的话,那么电源键的使用寿命一定不会长久. 虽然现在有很多的桌面软件都集成了一键锁屏,但是要安装这些软件就必须安装它们的其他组件,这点让人很不爽,因为他们会定期自动更新,或者在后台运行一些我不想要的东西.所以我决定自己写一个锁屏程序. 经过在android开发文档里面的查找,我发现

随机推荐