热门标签 | HotTags
当前位置:  开发笔记 > Android > 正文

Android使用MediaRecorder实现录音及播放

这篇文章主要为大家详细介绍了Android使用MediaRecorder实现录音及播放,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

现在项目中有使用到音视频相关技术,在参考了网上各种大牛的资料及根据自己项目实际情况(兼容安卓6.0以上版本动态权限管理等),把声音录制及播放相关代码做个记录。

public class MediaRecorderActivity extends BaseActivity {
 private Button start_tv;
 private ListView listView;
 //线程操作
 private ExecutorService mExecutorService;
 //录音API
 private MediaRecorder mMediaRecorder;
 //录音开始时间与结束时间
 private long startTime, endTime;
 //录音所保存的文件
 private File mAudioFile;
 //文件列表数据
 private List dataList;
 //录音文件数据列表适配器
 private AudioAdapter mAudioAdapter;
 //录音文件保存位置
 private String mFilePath = Environment.getExternalStorageDirectory().getAbsolutePath() + "/audio/";
 //当前是否正在播放
 private volatile boolean isPlaying;
 //播放音频文件API
 private MediaPlayer mediaPlayer;
 //使用Handler更新UI线程
 private Handler mHandler = new Handler() {
  @Override
  public void handleMessage(Message msg) {
   super.handleMessage(msg);
   switch (msg.what) {
    case Constant.RECORD_SUCCESS:
     //录音成功,展示数据
     if (null == mAudioAdapter) {
      mAudioAdapter = new AudioAdapter(MediaRecorderActivity.this, dataList, R.layout.file_item_layout);
     }
     listView.setAdapter(mAudioAdapter);
     break;
    //录音失败
    case Constant.RECORD_FAIL:
     showToastMsg(getString(R.string.record_fail));
     break;
    //录音时间太短
    case Constant.RECORD_TOO_SHORT:
     showToastMsg(getString(R.string.time_too_short));
     break;
    case Constant.PLAY_COMPLETION:
     showToastMsg(getString(R.string.play_over));
     break;
    case Constant.PLAY_ERROR:
     showToastMsg(getString(R.string.play_error));
     break;

   }
  }
 };

 @Override
 protected void setWindowView() {
  setContentView(R.layout.activity_record);
  //录音及播放要使用单线程操作
  mExecutorService = Executors.newSingleThreadExecutor();
  dataList = new ArrayList<>();
 }

 @Override
 protected void initViews() {
  this.start_tv = (Button) findViewById(R.id.start_tv);
  this.listView = (ListView) findViewById(R.id.listview);
 }

 @Override
 protected void initEvents() {
  //类似微信等应用按住说话进行录音,所以用OnTouch事件
  this.start_tv.setOnTouchListener(new View.OnTouchListener() {
   @Override
   public boolean onTouch(View view, MotionEvent motionEvent) {
    switch (motionEvent.getAction()) {
     //按下操作
     case MotionEvent.ACTION_DOWN:
      //安卓6.0以上录音相应权限处理
      if (Build.VERSION.SDK_INT > 22) {
       permissionForM();
      } else {
       startRecord();
      }
      break;
     //松开操作
     case MotionEvent.ACTION_CANCEL:
     case MotionEvent.ACTION_UP:
      stopRecord();
      break;
    }
    //对OnTouch事件做了处理,返回true
    return true;
   }
  });
  //点击播放对应的录音文件
  this.listView.setOnItemClickListener(new AdapterView.OnItemClickListener() {
   @Override
   public void onItemClick(AdapterView<&#63;> adapterView, View view, int i, long l) {
    //使用MediaPlayer播放声音文件
    startPlay(dataList.get(i).getFile());
   }
  });
 }


 /**
  * @description 开始进行录音
  * @author ldm
  * @time 2017/2/9 9:18
  */
 private void startRecord() {
  start_tv.setText(R.string.stop_by_up);
  start_tv.setBackgroundResource(R.drawable.bg_gray_round);
  //异步任务执行录音操作
  mExecutorService.submit(new Runnable() {
   @Override
   public void run() {
    //播放前释放资源
    releaseRecorder();
    //执行录音操作
    recordOperation();
   }
  });
 }

 /**
  * @description 录音失败处理
  * @author ldm
  * @time 2017/2/9 9:35
  */
 private void recordFail() {
  mAudioFile = null;
  mHandler.sendEmptyMessage(Constant.RECORD_FAIL);
 }

 /**
  * @description 录音操作
  * @author ldm
  * @time 2017/2/9 9:34
  */
 private void recordOperation() {
  //创建MediaRecorder对象
  mMediaRecorder = new MediaRecorder();
  //创建录音文件,.m4a为MPEG-4音频标准的文件的扩展名
  mAudioFile = new File(mFilePath + System.currentTimeMillis() + ".m4a");
  //创建父文件夹
  mAudioFile.getParentFile().mkdirs();
  try {
   //创建文件
   mAudioFile.createNewFile();
   //配置mMediaRecorder相应参数
   //从麦克风采集声音数据
   mMediaRecorder.setAudioSource(MediaRecorder.AudioSource.MIC);
   //设置保存文件格式为MP4
   mMediaRecorder.setOutputFormat(MediaRecorder.OutputFormat.MPEG_4);
   //设置采样频率,44100是所有安卓设备都支持的频率,频率越高,音质越好,当然文件越大
   mMediaRecorder.setAudioSamplingRate(44100);
   //设置声音数据编码格式,音频通用格式是AAC
   mMediaRecorder.setAudioEncoder(MediaRecorder.AudioEncoder.AAC);
   //设置编码频率
   mMediaRecorder.setAudioEncodingBitRate(96000);
   //设置录音保存的文件
   mMediaRecorder.setOutputFile(mAudioFile.getAbsolutePath());
   //开始录音
   mMediaRecorder.prepare();
   mMediaRecorder.start();
   //记录开始录音时间
   startTime = System.currentTimeMillis();
  } catch (Exception e) {
   e.printStackTrace();
   recordFail();
  }
 }


 /**
  * @description 结束录音操作
  * @author ldm
  * @time 2017/2/9 9:18
  */
 private void stopRecord() {
  start_tv.setText(R.string.speak_by_press);
  start_tv.setBackgroundResource(R.drawable.bg_white_round);
  //停止录音
  mMediaRecorder.stop();
  //记录停止时间
  endTime = System.currentTimeMillis();
  //录音时间处理,比如只有大于2秒的录音才算成功
  int time = (int) ((endTime - startTime) / 1000);
  if (time >= 3) {
   //录音成功,添加数据
   FileBean bean = new FileBean();
   bean.setFile(mAudioFile);
   bean.setFileLength(time);
   dataList.add(bean);
   //录音成功,发Message
   mHandler.sendEmptyMessage(Constant.RECORD_SUCCESS);
  } else {
   mAudioFile = null;
   mHandler.sendEmptyMessage(Constant.RECORD_TOO_SHORT);
  }
  //录音完成释放资源
  releaseRecorder();
 }

 /**
  * @description 翻放录音相关资源
  * @author ldm
  * @time 2017/2/9 9:33
  */
 private void releaseRecorder() {
  if (null != mMediaRecorder) {
   mMediaRecorder.release();
   mMediaRecorder = null;
  }
 }

 @Override
 public void onClick(View view) {

 }

 @Override
 protected void onDestroy() {
  super.onDestroy();
  //页面销毁,线程要关闭
  mExecutorService.shutdownNow();
 }
 /*******6.0以上版本手机权限处理***************************/
 /**
  * @description 兼容手机6.0权限管理
  * @author ldm
  * @time 2016/5/24 14:59
  */
 private void permissionForM() {
  if (ContextCompat.checkSelfPermission(this,
    Manifest.permission.RECORD_AUDIO) != PackageManager.PERMISSION_GRANTED || ContextCompat.checkSelfPermission(this,
    Manifest.permission.WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) {

   ActivityCompat.requestPermissions(this,
     new String[]{Manifest.permission.RECORD_AUDIO, Manifest.permission.WRITE_EXTERNAL_STORAGE},
     Constant.PERMISSIONS_REQUEST_FOR_AUDIO);
  } else {
   startRecord();
  }

 }

 @Override
 public void onRequestPermissionsResult(int requestCode,
           @NonNull String[] permissions, @NonNull int[] grantResults) {

  if (requestCode == Constant.PERMISSIONS_REQUEST_FOR_AUDIO) {
   if (grantResults[0] == PackageManager.PERMISSION_GRANTED) {
    startRecord();
   }
   return;
  }
  super.onRequestPermissionsResult(requestCode, permissions, grantResults);
 }

 /**
  * @description 播放音频
  * @author ldm
  * @time 2017/2/9 16:54
  */
 private void playAudio(final File mFile) {
  if (null != mFile && !isPlaying) {
   isPlaying = true;
   mExecutorService.submit(new Runnable() {
    @Override
    public void run() {
     startPlay(mFile);
    }
   });
  }
 }

 /**
  * @description 开始播放音频文件
  * @author ldm
  * @time 2017/2/9 16:56
  */
 private void startPlay(File mFile) {
  try {
   //初始化播放器
   mediaPlayer = new MediaPlayer();
   //设置播放音频数据文件
   mediaPlayer.setDataSource(mFile.getAbsolutePath());
   //设置播放监听事件
   mediaPlayer.setOnCompletionListener(new MediaPlayer.OnCompletionListener() {
    @Override
    public void onCompletion(MediaPlayer mediaPlayer) {
     //播放完成
     playEndOrFail(true);
    }
   });
   //播放发生错误监听事件
   mediaPlayer.setOnErrorListener(new MediaPlayer.OnErrorListener() {
    @Override
    public boolean onError(MediaPlayer mediaPlayer, int i, int i1) {
     playEndOrFail(false);
     return true;
    }
   });
   //播放器音量配置
   mediaPlayer.setVolume(1, 1);
   //是否循环播放
   mediaPlayer.setLooping(false);
   //准备及播放
   mediaPlayer.prepare();
   mediaPlayer.start();
  } catch (IOException e) {
   e.printStackTrace();
   //播放失败正理
   playEndOrFail(false);
  }

 }

 /**
  * @description 停止播放或播放失败处理
  * @author ldm
  * @time 2017/2/9 16:58
  */
 private void playEndOrFail(boolean isEnd) {
  isPlaying = false;
  if (isEnd) {
   mHandler.sendEmptyMessage(Constant.PLAY_COMPLETION);
  } else {
   mHandler.sendEmptyMessage(Constant.PLAY_ERROR);
  }
  if (null != mediaPlayer) {
   mediaPlayer.setOnCompletionListener(null);
   mediaPlayer.setOnErrorListener(null);
   mediaPlayer.stop();
   mediaPlayer.reset();
   mediaPlayer.release();
   mediaPlayer = null;
  }
 }
}

页面布局



 

对应资源文件strings.xml:


 mediarecorder
 录音失败
 时间太短,请重新录音
 播放完成
 抱歉,播放发生异常
 松开停止录音
 按住说话
 开始录音
 停止录音

录音相关权限 :





安卓录制播放音频:https://github.com/ldm520/Android_Media

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


推荐阅读
  • Java验证码——kaptcha的使用配置及样式
    本文介绍了如何使用kaptcha库来实现Java验证码的配置和样式设置,包括pom.xml的依赖配置和web.xml中servlet的配置。 ... [详细]
  • XML介绍与使用的概述及标签规则
    本文介绍了XML的基本概念和用途,包括XML的可扩展性和标签的自定义特性。同时还详细解释了XML标签的规则,包括标签的尖括号和合法标识符的组成,标签必须成对出现的原则以及特殊标签的使用方法。通过本文的阅读,读者可以对XML的基本知识有一个全面的了解。 ... [详细]
  • 本文介绍了Android 7的学习笔记总结,包括最新的移动架构视频、大厂安卓面试真题和项目实战源码讲义。同时还分享了开源的完整内容,并提醒读者在使用FileProvider适配时要注意不同模块的AndroidManfiest.xml中配置的xml文件名必须不同,否则会出现问题。 ... [详细]
  • Android中高级面试必知必会,积累总结
    本文介绍了Android中高级面试的必知必会内容,并总结了相关经验。文章指出,如今的Android市场对开发人员的要求更高,需要更专业的人才。同时,文章还给出了针对Android岗位的职责和要求,并提供了简历突出的建议。 ... [详细]
  • 本文讨论了Alink回归预测的不完善问题,指出目前主要针对Python做案例,对其他语言支持不足。同时介绍了pom.xml文件的基本结构和使用方法,以及Maven的相关知识。最后,对Alink回归预测的未来发展提出了期待。 ... [详细]
  • baresip android编译、运行教程1语音通话
    本文介绍了如何在安卓平台上编译和运行baresip android,包括下载相关的sdk和ndk,修改ndk路径和输出目录,以及创建一个c++的安卓工程并将目录考到cpp下。详细步骤可参考给出的链接和文档。 ... [详细]
  • 【Windows】实现微信双开或多开的方法及步骤详解
    本文介绍了在Windows系统下实现微信双开或多开的方法,通过安装微信电脑版、复制微信程序启动路径、修改文本文件为bat文件等步骤,实现同时登录两个或多个微信的效果。相比于使用虚拟机的方法,本方法更简单易行,适用于任何电脑,并且不会消耗过多系统资源。详细步骤和原理解释请参考本文内容。 ... [详细]
  • 安卓select模态框样式改变_微软Office风格的多端(Web、安卓、iOS)组件库——Fabric UI...
    介绍FabricUI是微软开源的一套Office风格的多端组件库,共有三套针对性的组件,分别适用于web、android以及iOS,Fab ... [详细]
  • Google Play推出全新的应用内评价API,帮助开发者获取更多优质用户反馈。用户每天在Google Play上发表数百万条评论,这有助于开发者了解用户喜好和改进需求。开发者可以选择在适当的时间请求用户撰写评论,以获得全面而有用的反馈。全新应用内评价功能让用户无需返回应用详情页面即可发表评论,提升用户体验。 ... [详细]
  • flowable工作流 流程变量_信也科技工作流平台的技术实践
    1背景随着公司业务发展及内部业务流程诉求的增长,目前信息化系统不能够很好满足期望,主要体现如下:目前OA流程引擎无法满足企业特定业务流程需求,且移动端体 ... [详细]
  • 在Android开发中,使用Picasso库可以实现对网络图片的等比例缩放。本文介绍了使用Picasso库进行图片缩放的方法,并提供了具体的代码实现。通过获取图片的宽高,计算目标宽度和高度,并创建新图实现等比例缩放。 ... [详细]
  • ZSI.generate.Wsdl2PythonError: unsupported local simpleType restriction ... [详细]
  • Android开发实现的计时器功能示例
    本文分享了Android开发实现的计时器功能示例,包括效果图、布局和按钮的使用。通过使用Chronometer控件,可以实现计时器功能。该示例适用于Android平台,供开发者参考。 ... [详细]
  • r2dbc配置多数据源
    R2dbc配置多数据源问题根据官网配置r2dbc连接mysql多数据源所遇到的问题pom配置可以参考官网,不过我这样配置会报错我并没有这样配置将以下内容添加到pom.xml文件d ... [详细]
  • 本文是关于自学Android的笔记,包括查看类的源码的方法,活动注册的必要性以及布局练习的重要性。通过学习本文,读者可以了解到在自学Android过程中的一些关键点和注意事项。 ... [详细]
author-avatar
漆黑中的萤火虫
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有