編輯:關於Android編程
Android 錄音與播放功能的簡單實例
最近在研究Android中一些常用的功能,像地圖、拍照、錄音和播放的實現等等,還有一些側滑、動畫等是如何實現的。
今天就把錄音和播放的實現分享一下,錄音和播放比較簡單,利用android內部的類即可實現。
1、先看下運行後的界面:
以下三張圖分別是進入、錄音、播放時的。



2、Layout布局文件
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="fill_parent"
android:layout_height="fill_parent" >
<TextView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_alignParentTop="true"
android:background="@drawable/switcherbar_bg"
android:gravity="center"
android:text="@string/audio_record_title"
android:textColor="#ffffff"
android:textSize="16sp" />
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_centerInParent="true"
android:orientation="vertical"
android:paddingLeft="10dip"
android:paddingRight="10dip" >
<TextView
android:id="@+id/audio_record_time"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="00:00" />
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginTop="10dip"
android:orientation="horizontal" >
<Button
android:id="@+id/audio_record_start"
android:layout_width="0dip"
android:layout_height="wrap_content"
android:layout_weight="1"
android:background="@drawable/searchbtn_normal"
android:text="開始錄音"
android:textSize="14sp" />
<Button
android:id="@+id/audio_record_stop"
android:layout_width="0dip"
android:layout_height="wrap_content"
android:layout_marginLeft="5dip"
android:layout_weight="1"
android:background="@drawable/searchbtn_bg"
android:text="結束錄音"
android:textSize="14sp" />
</LinearLayout>
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginTop="10dip"
android:orientation="horizontal" >
<Button
android:id="@+id/audio_record_play"
android:layout_width="0dip"
android:layout_height="wrap_content"
android:layout_weight="1"
android:background="@drawable/searchbtn_normal"
android:text="播放錄音"
android:textSize="14sp" />
<Button
android:id="@+id/audio_record_select"
android:layout_width="0dip"
android:layout_height="wrap_content"
android:layout_marginLeft="5dip"
android:layout_weight="1"
android:background="@drawable/searchbtn_bg"
android:text="確定選擇"
android:textSize="14sp" />
</LinearLayout>
</LinearLayout>
</RelativeLayout>
3 Activity類
錄音涉及到二個Activity,第一個Activity比較簡單,我這裡大概說下,其實就是有個按鈕,點擊後轉移第二個Activity,錄音返回後,在第一個Activity中獲取錄音的文件名、時長等。
第一個Activity部分代碼:
// 錄音事件
ksly_btn.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
Intent intent = new Intent(MaintainVisitEditActivity.this, AudioRecordActivity.class);
intent.setAction(Intent.ACTION_VIEW);
intent.putExtra("duration", entity.getVoiceDuration());
intent.putExtra("fileName", entity.getVoiceRecord());
startActivityForResult(intent, VOICE_RECODE);
}
});
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
super.onActivityResult(requestCode, resultCode, data);
if (requestCode == VOICE_RECODE && resultCode == AudioRecordActivity.SUCCESS) {
entity.setVoiceDuration(data.getLongExtra("duration", 0));// 時長
entity.setVoiceRecord(data.getStringExtra("fileName"));// 文件名(絕對路徑)
ksly_time.setText(DateTimeUtils.formatToMillisecond(entity.getVoiceDuration()));
}
}
第二個Activity代碼:
這裡要注意一下,就是需要捕獲返回鍵,處理一下,就是點擊返回鍵時,也返回個狀態碼,以表示沒有錄音成功。
package com.whowii.ct.cm.activity;
import java.io.File;
import java.io.IOException;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.Timer;
import java.util.TimerTask;
import android.app.Activity;
import android.content.Intent;
import android.media.MediaPlayer;
import android.media.MediaPlayer.OnCompletionListener;
import android.media.MediaRecorder;
import android.os.Bundle;
import android.os.Handler;
import android.os.Message;
import android.util.Log;
import android.view.KeyEvent;
import android.view.View;
import android.view.View.OnClickListener;
import android.view.Window;
import android.widget.Button;
import android.widget.TextView;
import android.widget.Toast;
import com.whowii.ct.cm.R;
import com.whowii.ct.cm.command.QueryParams;
import com.whowii.ct.cm.utils.DateTimeUtils;
import com.whowii.ct.cm.utils.SDCardUtils;
/**
* 錄制音頻
*
* @author Administrator
*
*/
public class AudioRecordActivity extends Activity {
private TextView audio_record_time;
private Button audio_record_start, audio_record_stop, audio_record_play, audio_record_select;
private MediaRecorder mediaRecorder;
private final String TAG = AudioRecordActivity.class.getSimpleName();
private boolean isIdle = true;// 當前是否空閒,false:表示正在錄音
private long startTime = 0, stopTime = 0, duration = 0;// 開始時間、結束時間、錄音時長
private String fileName = null;// 存儲錄音文件的路徑
private Timer timer = null;// Timer計時器
public static final int SUCCESS = 1;// 錄制成功;
public static final int FAILURE = 0;// 錄制失敗
private MediaPlayer mediaPlayer;
private TimerTask task = new TimerTask() {
final Handler handler = new Handler() {
public void handleMessage(Message message) {
Bundle data = message.getData();
audio_record_time.setText(DateTimeUtils.formatToMillisecond(data.getLong("time")));
}
};
public void run() {
Message message = new Message();
long t = System.currentTimeMillis();
Bundle data = new Bundle();
data.putLong("time", t - startTime);
message.setData(data);
handler.sendMessage(message);
}
};
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
requestWindowFeature(Window.FEATURE_NO_TITLE); // 設置無標題欄
setContentView(R.layout.audio_record);
mediaPlayer = new MediaPlayer();
initControl();
setListener();
timer = new Timer(true);
fileName = getIntent().getStringExtra("fileName");
duration = getIntent().getLongExtra("duration", 0);
}
private void initControl() {
audio_record_time = (TextView) findViewById(R.id.audio_record_time);
audio_record_start = (Button) findViewById(R.id.audio_record_start);
audio_record_stop = (Button) findViewById(R.id.audio_record_stop);
audio_record_play = (Button) findViewById(R.id.audio_record_play);
audio_record_select = (Button) findViewById(R.id.audio_record_select);
}
private void setListener() {
// 播放完成事件
mediaPlayer.setOnCompletionListener(new OnCompletionListener() {
@Override
public void onCompletion(MediaPlayer mp) {
isIdle = true;
audio_record_play.setText("播放錄音");
audio_record_play.setBackgroundResource(R.drawable.searchinput_bg);
}
});
// 開始錄音
audio_record_start.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
if (!isIdle) {
return;
}
if (!SDCardUtils.sdCardExists()) {
Toast.makeText(AudioRecordActivity.this, "缺少SD卡,請先插入後再操作!", Toast.LENGTH_LONG).show();
return;
}
audio_record_start.setText("開始錄音");
audio_record_start.setEnabled(true);
duration = 0;
startTime = System.currentTimeMillis();
fileName = QueryParams.CACHE_AUDIO_PATH;
fileName += new SimpleDateFormat("yyyyMMdd_hhmmss").format(new Date(startTime)) + ".amr";
File file = new File(fileName);
mediaRecorder = new MediaRecorder();
mediaRecorder.setOutputFile(file.getAbsolutePath());
mediaRecorder.setAudioSource(MediaRecorder.AudioSource.MIC);
mediaRecorder.setOutputFormat(MediaRecorder.OutputFormat.THREE_GPP);
mediaRecorder.setAudioEncoder(MediaRecorder.AudioEncoder.AMR_NB);
try {
mediaRecorder.prepare();
mediaRecorder.start();
isIdle = false;
audio_record_start.setBackgroundResource(R.drawable.searchbtn_pressed);
timer.schedule(task, 0, 100);
} catch (IOException e) {
startTime = 0;
Log.e(TAG, e.toString());
Toast.makeText(AudioRecordActivity.this, "錄制時發生異常!", Toast.LENGTH_LONG).show();
}
}
});
// 結束錄音
audio_record_stop.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
if (mediaRecorder != null) {
stopTime = System.currentTimeMillis();
duration = stopTime - startTime;
timer.cancel();
mediaRecorder.stop();
mediaRecorder.release();
mediaRecorder = null;
audio_record_start.setBackgroundResource(R.drawable.searchbtn_normal);
isIdle = true;
}
}
});
// 播放錄音
audio_record_play.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
if (!isIdle) {
return;
}
if (audio_record_play.getText().equals("播放錄音")) {
if (fileName == null || fileName.equals("") || duration == 0) {
Toast.makeText(AudioRecordActivity.this, "沒有錄音文件!", Toast.LENGTH_LONG).show();
return;
}
try {
mediaPlayer.reset();
mediaPlayer.setDataSource(fileName);
mediaPlayer.prepare();
mediaPlayer.start();
isIdle = false;
audio_record_play.setText("終止播放");
audio_record_play.setBackgroundResource(R.drawable.searchbtn_pressed);
} catch (Exception e) {
e.printStackTrace();
Toast.makeText(AudioRecordActivity.this, "播放錄音時遇到錯誤!", Toast.LENGTH_LONG).show();
}
} else {
if (mediaPlayer != null && mediaPlayer.isPlaying()) {
mediaPlayer.stop();
isIdle = true;
}
audio_record_play.setText("播放錄音");
audio_record_play.setBackgroundResource(R.drawable.searchinput_bg);
}
}
});
// 確認選擇
audio_record_select.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
if (fileName == null || fileName.equals("") || duration == 0) {
Toast.makeText(AudioRecordActivity.this, "沒有錄音文件!", Toast.LENGTH_LONG).show();
return;
}
Intent intent = new Intent();
intent.putExtra("fileName", fileName);
intent.putExtra("duration", duration);
setResult(SUCCESS, intent);// 返回成功標識
isIdle = true;
if (mediaPlayer != null) {
if (mediaPlayer.isPlaying()) {
mediaPlayer.stop();
}
mediaPlayer = null;
}
finish();// 結束當前的activity,等於點擊返回按鈕
}
});
}
// 捕獲返回鍵,關閉當前頁面時返回失敗標識
@Override
public boolean onKeyDown(int keyCode, KeyEvent event) {
if (keyCode == KeyEvent.KEYCODE_BACK) {
setResult(FAILURE);
isIdle = true;
if (mediaPlayer != null) {
if (mediaPlayer.isPlaying()) {
mediaPlayer.stop();
}
mediaPlayer = null;
}
finish();
return true;
}
return super.onKeyDown(keyCode, event);
}
}
感謝閱讀,希望能幫助到大家,謝謝大家對本站的支持!
《Android源碼設計模式解析與實戰》讀書筆記(十八)
第十八章、代理模式 代理模式也稱委托模式,是結構型設計模式之一。是應用廣泛的模式之一。1.定義為其他對象提供一種代理以控制對這個對象的訪問。2.使用場景當無法或不想直接訪
android異步加載
本篇博客總結了慕課網關於異步加載圖片的知識要點,和大家一起分享,有感覺聽得不連貫的可以來看看。看完本篇博客,你將學習到下面的知識:1.怎樣將一個url(也可以說是一個In
Android基礎入門教程——8.4.2 Android動畫合集之補間動畫
本節引言: 本節帶來的是Android三種動畫中的第二種——補間動畫(Tween),和前面學的幀動畫不同,幀動畫 是通過連續播放圖片來
淺析安卓(Android)的性能優化
Android性能的優化主要分為兩點1、布局優化2、內存優化布局優化首先來看一下布局優化,系統在渲染UI的時候會消耗大量的資源,所以,對布局的優化就顯得尤為重要避免Ove