Android使用MediaRecorder实现录音及播放
发布时间 - 2026-01-10 23:01:11 点击率:次现在项目中有使用到音视频相关技术,在参考了网上各种大牛的资料及根据自己项目实际情况(兼容安卓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<FileBean> 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<?> 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;
}
}
}
页面布局
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="vertical" android:paddingLeft="@dimen/activity_horizontal_margin" android:paddingRight="@dimen/activity_horizontal_margin" android:paddingTop="@dimen/activity_vertical_margin" android:paddingBottom="@dimen/activity_vertical_margin"> <Button android:id="@+id/start_tv" android:layout_width="match_parent" android:layout_height="wrap_content" android:text="开始录音" android:textSize="16sp" /> <ListView android:id="@+id/listview" android:layout_width="match_parent" android:layout_height="wrap_content" android:divider="#d1d1d1" android:dividerHeight="1dp" android:scrollbars="none" android:layout_marginTop="10dp" android:layout_marginBottom="10dp"></ListView> </LinearLayout>
对应资源文件strings.xml:
<resources> <string name="app_name">mediarecorder</string> <string name="record_fail">录音失败</string> <string name="time_too_short">时间太短,请重新录音</string> <string name="play_over">播放完成</string> <string name="play_error">抱歉,播放发生异常</string> <string name="stop_by_up">松开停止录音</string> <string name="speak_by_press">按住说话</string> <string name="start_record">开始录音</string> <string name="stop_record">停止录音</string> </resources>
录音相关权限 :
<!--SD卡权限--> <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/> <!--录音权限--> <uses-permission android:name="android.permission.RECORD_AUDIO"/>
安卓录制播放音频:https://github.com/ldm520/Android_Media
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。
# Android
# MediaRecorder
# 录音
# android MediaRecorder实现录屏时带录音功能
# Android编程录音工具类RecorderUtil定义与用法示例
# Android实现录音功能实现实例(MediaRecorder)
# Android使用AudioRecord实现暂停录音功能实例代码
# Android录音--AudioRecord、MediaRecorder的使用
# android 通过MediaRecorder实现简单的录音示例
# Android App调用MediaRecorder实现录音功能的实例
# Android音频录制MediaRecorder之简易的录音软件实现代码
# Android简单的利用MediaRecorder进行录音的实例代码
# Android使用AudioRecord实现录音功能
# 播放器
# 太短
# 中有
# 扩展名
# 做个
# 实际情况
# 越好
# 按下
# 越大
# 文件列表
# 越高
# 正在播放
# 才算
# 音视频
# 要使
# 大家多多
# 发生错误
# 单线程
# 格式为
# 大牛
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
米侠浏览器网页图片不显示怎么办 米侠图片加载修复
Laravel怎么创建控制器Controller_Laravel路由绑定与控制器逻辑编写【指南】
如何用腾讯建站主机快速创建免费网站?
如何在服务器上配置二级域名建站?
制作无缝贴图网站有哪些,3dmax无缝贴图怎么调?
Laravel如何发送邮件和通知_Laravel邮件与通知系统发送步骤
标准网站视频模板制作软件,现在有哪个网站的视频编辑素材最齐全的,背景音乐、音效等?
Laravel用户密码怎么加密_Laravel Hash门面使用教程
简历在线制作网站免费版,如何创建个人简历?
如何快速查询域名建站关键信息?
iOS正则表达式验证手机号、邮箱、身份证号等
Laravel如何集成第三方登录_Laravel Socialite实现微信QQ微博登录
微信小程序制作网站有哪些,微信小程序需要做网站吗?
laravel怎么在请求结束后执行任务(Terminable Middleware)_laravel Terminable Middleware请求结束任务执行方法
西安市网站制作公司,哪个相亲网站比较好?西安比较好的相亲网站?
Laravel全局作用域是什么_Laravel Eloquent Global Scopes应用指南
laravel怎么为应用开启和关闭维护模式_laravel应用维护模式开启与关闭方法
如何在万网开始建站?分步指南解析
微信小程序 五星评分(包括半颗星评分)实例代码
敲碗10年!Mac系列传将迎来「触控与联网」双革新
Android自定义控件实现温度旋转按钮效果
Zeus浏览器网页版官网入口 宙斯浏览器官网在线通道
如何用虚拟主机快速搭建网站?详细步骤解析
ChatGPT怎么生成Excel公式_ChatGPT公式生成方法【指南】
如何在香港免费服务器上快速搭建网站?
Laravel怎么实现验证码(Captcha)功能
轻松掌握MySQL函数中的last_insert_id()
详解一款开源免费的.NET文档操作组件DocX(.NET组件介绍之一)
如何生成腾讯云建站专用兑换码?
大连 网站制作,大连天途有线官网?
公司门户网站制作公司有哪些,怎样使用wordpress制作一个企业网站?
Laravel如何优雅地处理服务层_在Laravel中使用Service层和Repository层
详解Nginx + Tomcat 反向代理 如何在高效的在一台服务器部署多个站点
深圳网站制作平台,深圳市做网站好的公司有哪些?
Laravel如何使用Seeder填充数据_Laravel模型工厂Factory批量生成测试数据【方法】
bing浏览器学术搜索入口_bing学术文献检索地址
学生网站制作软件,一个12岁的学生写小说,应该去什么样的网站?
开心动漫网站制作软件下载,十分开心动画为何停播?
Windows10如何更改计算机工作组_Win10系统属性修改Workgroup
Android实现代码画虚线边框背景效果
Laravel如何优化应用性能?(缓存和优化命令)
Laravel怎么配置S3云存储驱动_Laravel集成阿里云OSS或AWS S3存储桶【教程】
佐糖AI抠图怎样调整抠图精度_佐糖AI精度调整与放大细化操作【攻略】
php在windows下怎么调试_phpwindows环境调试操作说明【操作】
如何打造高效商业网站?建站目的决定转化率
UC浏览器如何切换小说阅读源_UC浏览器阅读源切换【方法】
Python数据仓库与ETL构建实战_Airflow调度流程详解
Laravel如何记录自定义日志?(Log频道配置)
如何在局域网内绑定自建网站域名?
javascript中数组(Array)对象和字符串(String)对象的常用方法总结

