Android實現(xiàn)錄音功能實現(xiàn)實例(MediaRecorder)
本文介紹了Android實現(xiàn)錄音的實例代碼(MediaRecorder),分享給大家,具體如下:
Android提供了兩個API用于錄音的實現(xiàn):MediaRecorder 和 AudioRecord,各有優(yōu)劣。
1、MediaRecorder
已經(jīng)集成了錄音、編碼、壓縮等,支持少量的錄音音頻格式,大概有.aac(API = 16) .amr .3gp
優(yōu)點:大部分已經(jīng)集成,直接調(diào)用相關(guān)接口即可,代碼量小
缺點:無法實時處理音頻;輸出的音頻格式不是很多,例如沒有輸出mp3格式文件
2、AudioRecord
主要是實現(xiàn)邊錄邊播(AudioRecord+AudioTrack)以及對音頻的實時處理(如會說話的湯姆貓、語音)
優(yōu)點:語音的實時處理,可以用代碼實現(xiàn)各種音頻的封裝
缺點:輸出是PCM語音數(shù)據(jù),如果保存成音頻文件,是不能夠被播放器播放的,所以必須先寫代碼實現(xiàn)數(shù)據(jù)編碼以及壓縮
先說 MediaRecorder : MediaRecorder因為大部分功能已經(jīng)集成,所以使用起來相對比較簡單。
下面是個小demo:
① 界面
界面比較簡單,由于MediaRecorder 并不能實現(xiàn)暫停、繼續(xù)錄音的功能
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:id="@+id/activity_main" android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="vertical"> <Button android:id="@+id/btn_start" android:layout_width="match_parent" android:layout_height="wrap_content" android:text="開始" /> <Button android:id="@+id/btn_stop" android:layout_width="match_parent" android:layout_height="wrap_content" android:enabled="false" android:text="停止" /> <TextView android:id="@+id/text_time" android:layout_width="match_parent" android:layout_height="wrap_content" android:gravity="center" android:text="00:00:00" android:padding="5dp" android:layout_marginTop="10dp"/> </LinearLayout>
② 相關(guān)錄音功能
開始錄音
public void startRecord() { // 開始錄音 /* ①Initial:實例化MediaRecorder對象 */ if (mMediaRecorder == null) mMediaRecorder = new MediaRecorder(); try { /* ②setAudioSource/setVedioSource */ mMediaRecorder.setAudioSource(MediaRecorder.AudioSource.MIC);// 設(shè)置麥克風 /* * ②設(shè)置輸出文件的格式:THREE_GPP/MPEG-4/RAW_AMR/Default THREE_GPP(3gp格式 * ,H263視頻/ARM音頻編碼)、MPEG-4、RAW_AMR(只支持音頻且音頻編碼要求為AMR_NB) */ mMediaRecorder.setOutputFormat(MediaRecorder.OutputFormat.MPEG_4); /* ②設(shè)置音頻文件的編碼:AAC/AMR_NB/AMR_MB/Default 聲音的(波形)的采樣 */ mMediaRecorder.setAudioEncoder(MediaRecorder.AudioEncoder.AAC); fileName = DateFormat.format("yyyyMMdd_HHmmss", Calendar.getInstance(Locale.CHINA)) + ".m4a"; if (!FileUtils.isFolderExist(FileUtils.getFolderName(audioSaveDir))) { FileUtils.makeFolders(audioSaveDir); } filePath = audioSaveDir + fileName; /* ③準備 */ mMediaRecorder.setOutputFile(filePath); mMediaRecorder.prepare(); /* ④開始 */ mMediaRecorder.start(); } catch (IllegalStateException e) { LogUtil.i("call startAmr(File mRecAudioFile) failed!" + e.getMessage()); } catch (IOException e) { LogUtil.i("call startAmr(File mRecAudioFile) failed!" + e.getMessage()); } }
音頻編碼可以根據(jù)自己實際需要自己設(shè)定,文件名防止重復(fù),使用了日期_時分秒的結(jié)構(gòu),audioSaveDir 是文件存儲目錄,可自行設(shè)定。
停止錄音
public void stopRecord() { try { mMediaRecorder.stop(); mMediaRecorder.release(); mMediaRecorder = null; filePath = ""; } catch (RuntimeException e) { LogUtil.e(e.toString()); mMediaRecorder.reset(); mMediaRecorder.release(); mMediaRecorder = null; File file = new File(filePath); if (file.exists()) file.delete(); filePath = ""; } }
時長記錄
// 記錄錄音時長 private void countTime() { while (isRecording) { LogUtil.d("正在錄音"); timeCount++; Message msg = Message.obtain(); msg.what = TIME_COUNT; msg.obj = timeCount; myHandler.sendMessage(msg); try { timeThread.sleep(1000); } catch (InterruptedException e) { e.printStackTrace(); } } LogUtil.d("結(jié)束錄音"); timeCount = 0; Message msg = Message.obtain(); msg.what = TIME_COUNT; msg.obj = timeCount; myHandler.sendMessage(msg); }
將錄音時長格式化
// 格式化 錄音時長為 時:分:秒 public static String FormatMiss(int miss) { String hh = miss / 3600 > 9 ? miss / 3600 + "" : "0" + miss / 3600; String mm = (miss % 3600) / 60 > 9 ? (miss % 3600) / 60 + "" : "0" + (miss % 3600) / 60; String ss = (miss % 3600) % 60 > 9 ? (miss % 3600) % 60 + "" : "0" + (miss % 3600) % 60; return hh + ":" + mm + ":" + ss; }
Activity全部代碼
import android.media.MediaRecorder; import android.os.Bundle; import android.os.Environment; import android.os.Handler; import android.os.Message; import android.support.v7.app.AppCompatActivity; import android.text.format.DateFormat; import android.view.View; import android.widget.Button; import android.widget.TextView; import com.zzx.hellokotlin.R; import com.zzx.hellokotlin.utils.FileUtils; import com.zzx.hellokotlin.utils.LogUtil; import java.io.File; import java.io.IOException; import java.util.Calendar; import java.util.Locale; public class Record2Activity extends AppCompatActivity { // 錄音界面相關(guān) Button btnStart; Button btnStop; TextView textTime; // 錄音功能相關(guān) MediaRecorder mMediaRecorder; // MediaRecorder 實例 boolean isRecording; // 錄音狀態(tài) String fileName; // 錄音文件的名稱 String filePath; // 錄音文件存儲路徑 Thread timeThread; // 記錄錄音時長的線程 int timeCount; // 錄音時長 計數(shù) final int TIME_COUNT = 0x101; // 錄音文件存放目錄 final String audioSaveDir = Environment.getExternalStorageDirectory().getAbsolutePath()+"/audiodemo/"; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_record2); btnStart = (Button) findViewById(R.id.btn_start); btnStop = (Button) findViewById(R.id.btn_stop); textTime = (TextView) findViewById(R.id.text_time); btnStart.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { // 開始錄音 btnStart.setEnabled(false); btnStop.setEnabled(true); startRecord(); isRecording = true; // 初始化錄音時長記錄 timeThread = new Thread(new Runnable() { @Override public void run() { countTime(); } }); timeThread.start(); } }); btnStop.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { // 停止錄音 btnStart.setEnabled(true); btnStop.setEnabled(false); stopRecord(); isRecording = false; } }); } // 記錄錄音時長 private void countTime() { while (isRecording) { LogUtil.d("正在錄音"); timeCount++; Message msg = Message.obtain(); msg.what = TIME_COUNT; msg.obj = timeCount; myHandler.sendMessage(msg); try { timeThread.sleep(1000); } catch (InterruptedException e) { e.printStackTrace(); } } LogUtil.d("結(jié)束錄音"); timeCount = 0; Message msg = Message.obtain(); msg.what = TIME_COUNT; msg.obj = timeCount; myHandler.sendMessage(msg); } /** * 開始錄音 使用amr格式 * 錄音文件 * @return */ public void startRecord() { // 開始錄音 /* ①Initial:實例化MediaRecorder對象 */ if (mMediaRecorder == null) mMediaRecorder = new MediaRecorder(); try { /* ②setAudioSource/setVedioSource */ mMediaRecorder.setAudioSource(MediaRecorder.AudioSource.MIC);// 設(shè)置麥克風 /* * ②設(shè)置輸出文件的格式:THREE_GPP/MPEG-4/RAW_AMR/Default THREE_GPP(3gp格式 * ,H263視頻/ARM音頻編碼)、MPEG-4、RAW_AMR(只支持音頻且音頻編碼要求為AMR_NB) */ mMediaRecorder.setOutputFormat(MediaRecorder.OutputFormat.MPEG_4); /* ②設(shè)置音頻文件的編碼:AAC/AMR_NB/AMR_MB/Default 聲音的(波形)的采樣 */ mMediaRecorder.setAudioEncoder(MediaRecorder.AudioEncoder.AAC); fileName = DateFormat.format("yyyyMMdd_HHmmss", Calendar.getInstance(Locale.CHINA)) + ".m4a"; if (!FileUtils.isFolderExist(FileUtils.getFolderName(audioSaveDir))) { FileUtils.makeFolders(audioSaveDir); } filePath = audioSaveDir + fileName; /* ③準備 */ mMediaRecorder.setOutputFile(filePath); mMediaRecorder.prepare(); /* ④開始 */ mMediaRecorder.start(); } catch (IllegalStateException e) { LogUtil.i("call startAmr(File mRecAudioFile) failed!" + e.getMessage()); } catch (IOException e) { LogUtil.i("call startAmr(File mRecAudioFile) failed!" + e.getMessage()); } } /** * 停止錄音 */ public void stopRecord() { //有一些網(wǎng)友反應(yīng)在5.0以上在調(diào)用stop的時候會報錯,翻閱了一下谷歌文檔發(fā)現(xiàn)上面確實寫的有可能會報錯的情況,捕獲異常清理一下就行了,感謝大家反饋! try { mMediaRecorder.stop(); mMediaRecorder.release(); mMediaRecorder = null; filePath = ""; } catch (RuntimeException e) { LogUtil.e(e.toString()); mMediaRecorder.reset(); mMediaRecorder.release(); mMediaRecorder = null; File file = new File(filePath); if (file.exists()) file.delete(); filePath = ""; } } // 格式化 錄音時長為 時:分:秒 public static String FormatMiss(int miss) { String hh = miss / 3600 > 9 ? miss / 3600 + "" : "0" + miss / 3600; String mm = (miss % 3600) / 60 > 9 ? (miss % 3600) / 60 + "" : "0" + (miss % 3600) / 60; String ss = (miss % 3600) % 60 > 9 ? (miss % 3600) % 60 + "" : "0" + (miss % 3600) % 60; return hh + ":" + mm + ":" + ss; } Handler myHandler = new Handler() { @Override public void handleMessage(Message msg) { switch (msg.what) { case TIME_COUNT: int count = (int) msg.obj; LogUtil.d("count == " + count); textTime.setText(FormatMiss(count)); break; } } }; }
總結(jié):MediaRecorder 實現(xiàn)錄音還是比較簡單的,只是不能暫停。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習有所幫助,也希望大家多多支持腳本之家。
- Android給通知channel靜音的方法實例
- Android實現(xiàn)靜音檢測功能
- Android 判斷網(wǎng)絡(luò)狀態(tài)對音頻靜音的實現(xiàn)方法
- Android實現(xiàn)定時自動靜音小助手
- Android EasyPlayer聲音自動停止、恢復(fù),一鍵靜音等功能
- android實現(xiàn)來電靜音示例(監(jiān)聽來電)
- android系統(tǒng)在靜音模式下關(guān)閉camera拍照聲音的方法
- Android音頻錄制MediaRecorder之簡易的錄音軟件實現(xiàn)代碼
- Android簡單的利用MediaRecorder進行錄音的實例代碼
- Android實現(xiàn)錄音靜音降噪
相關(guān)文章
Android?雙屏異顯自適應(yīng)Dialog的實現(xiàn)
Android 多屏互聯(lián)的時代,必然會出現(xiàn)多屏連接的問題,本文主要介紹了Android?雙屏異顯自適應(yīng)Dialog的實現(xiàn),具有一定的參考價值,感興趣的可以了解一下2023-12-12Android 使用jarsigner給apk簽名的方法詳細介紹
這篇文章主要介紹了Android 使用jarsigner給apk簽名的方法詳細介紹的相關(guān)資料,APP 完成需要在一些APP 商店進行上傳審核,供用戶下載使用,APP 需要簽名認證,需要的朋友可以參考下2016-12-12ProtoBuf動態(tài)拆分Gradle?Module解析
這篇文章主要為大家介紹了ProtoBuf動態(tài)拆分Gradle?Module解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-02-02Android中監(jiān)聽Home鍵的4種方法總結(jié)
這篇文章主要介紹了Android中監(jiān)聽Home鍵的4種方法總結(jié),本文講解了onSaveInstanceState方法、onUserLeaveHint方法、ACTION_CLOSE_SYSTEM_DIALOGS、framework PhoneWindowManager.java等4種方法,需要的朋友可以參考下2015-04-04Android使用CircleImageView實現(xiàn)圓形頭像的方法
圓形頭像看起來非常美觀,下文通過實例代碼給大家介紹android中使用CircleImageView實現(xiàn)圓形頭像的方法,一起看看吧2016-09-09Android開發(fā)之在程序中時時獲取logcat日志信息的方法(附demo源碼下載)
這篇文章主要介紹了Android開發(fā)之在程序中時時獲取logcat日志信息的方法,結(jié)合實例形式較為詳細的分析了實時獲取logcat日志的原理、步驟與相關(guān)實現(xiàn)技巧,并附帶相應(yīng)的demo源碼供讀者下載參考,需要的朋友可以參考下2016-02-02