http://blog.csdn.net/java_android_c/article/details/52619737html
Android 音頻簡介java
常見的音頻編解碼的類型:AAC OPUS MP3 AMR Ogg PCM android
AAC: 高級音頻編碼 對應 .m4a(audio/m4a)或者.3pg(audio/3gpp)文件 HEAAC:高級AAC,使用的比較多。網絡
OPUS:有損聲音編碼的格式,由互聯網工程任務組(IETF)進來開發,適用於網絡上的實時聲音傳輸,如:語音通話ide
MP3: 使用的最普遍的音頻編解碼器 對應 .mp3(audio/mp3) 各類音樂網站通常用這種。佈局
AMR:自適應多速率音頻編解碼器,通常用於語音呼叫程序。網站
Ogg:開發的無專利的音頻編解碼器,其質量可與商業的和手專利保護的MP3以及AAC編解碼相媲美。this
PCM :原始音頻,在android平臺上由audio record直接錄用下來的,未通過編碼的。編碼
視頻直播,語音通話中通常使用AAC或者OPUS ,若是對聲音要進行處理就須要使用PCM原始音頻加工處理,而後再進行編碼.url
Android音頻採集(捕獲)
android平臺上的音頻採集通常就三種:1.利用android內置的應用程序 2.使用MediaRecorder進行音頻捕獲 3.使用AudioRecord進行音頻捕獲。此3種方式的靈活性逐漸增大,相應的所須要作的工做也逐漸增多。
1、Android 內置的應用程序。
- Intent intent=new Intent(MediaStore.Audio.Media.RECORD_SOUND_ACTION);
- startActivityForResult(intent,0);
這種方式靈活度最差,通常就是作着演示下,開發中基本不用這種方案。
2、使用MediaRecorder進行音頻的捕獲。 這個是 錄影視頻和音頻分別編碼後保存成一個文件,單獨音頻也能夠,不過是編碼後的數據
這種方案相較於調用系統內置的用用程序,靈活度要高不少,便於開發者在UI界面上佈局,並且系統封裝的很好,便於使用,惟一的缺點是使用它錄下來的音頻是通過編碼的,沒有辦法的獲得原始的音頻。同時MediaRecorder便可用於音頻的捕獲也能夠用於視頻的捕獲至關的強大。實際開發中沒有特殊需求的話,用的是比較多的!
使用步驟:
建立MediaRecorder對象,調用以下方法(Ps:調用順序順序對結果的影響是很是的大。)
MediaRecorder recorder=new MediaRecorder();//建立MediaRecoder對象
1.recorder.setAudioSource(MediaRecorder.AudioSource.MIC); //調用setAudioSource方法 (調用的第一個方法)
MediaRecorder.AudioSource.CAMCORDER和MediaRecorder.AudioSource.VOICE_RECOGNITION當設備具備。>=2個麥克風的時候就可使用它們。
MediaRecorder.AudioSource.VOICE_CALL從電話中錄音
2.recorder.setOutputFormat(MediaRecorder.OutputFormat.THREE_GPP);//setOutputFormat方法(調用的第二個方法)
MediaRecorder.OutputFormat.THREE_GPP 輸出文件將是一個擴展名爲(.3gp)的文件。它可能包含音頻和視頻。
MediaRecorder.OutputFormat.MPEG_4 輸出一個MPEG_4文件,它可能包含音頻和視頻。
3.recorder.setAudioEncoder(MediaRecorder.AudioEncoder.AAC); //setAudioEncoder方法 (這是調用的第三個方法)
4.recorder.setOutputFile(url); //setOutputFile方法 url是目標文件路徑(這是調用的第四個方法)
以上四個方法一次調用完成以後,就能夠像MediaPlayer同樣控制錄製和暫停錄製了。
- <?xml version="1.0" encoding="utf-8"?>
- <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
- android:orientation="vertical"
- android:layout_width="fill_parent"
- android:layout_height="fill_parent"
- android:background="#fff"
- android:gravity="center">
-
- <Button
- android:text="開始錄音"
- android:id="@+id/StartRecording"
- android:layout_width="wrap_content"
- android:layout_height="wrap_content"/>
-
- <Button
- android:text="中止錄音"
- android:id="@+id/StopRecording"
- android:layout_width="wrap_content"
- android:layout_height="wrap_content"/>
-
- </LinearLayout>
- public class MainActivity extends Activity implements OnClickListener{
-
- Button startRecording, stopRecording;
- MediaRecorder recorder;
- File audioFile;
- boolean isRecoding=false;
-
- @Override
- public void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- requestWindowFeature(Window.FEATURE_NO_TITLE);
- setContentView(R.layout.main);
- init();
- initListener();
- }
-
-
- public void init(){
- stopRecording = (Button) this.findViewById(R.id.StopRecording);
- startRecording = (Button) this.findViewById(R.id.StartRecording);
-
- recorder = new MediaRecorder();
- recorder.setAudioSource(MediaRecorder.AudioSource.MIC);
- recorder.setOutputFormat(MediaRecorder.OutputFormat.THREE_GPP);
- recorder.setAudioEncoder(MediaRecorder.AudioEncoder.AMR_NB);
- }
-
-
- public void initListener(){
- startRecording.setOnClickListener(this);
- stopRecording.setOnClickListener(this);
- }
-
-
- public void recod(){
-
-
-
- File path = new File(Environment.getExternalStorageDirectory().getAbsolutePath()+ "/MediaRecorderTest");
- if(!path.exists())
- {
- path.mkdirs();
- }
-
- try {
- audioFile=new File(path,"test.3gp");
- if(audioFile.exists())
- {
- audioFile.delete();
- }
- audioFile.createNewFile();
-
- } catch (Exception e) {
- throw new RuntimeException("Couldn't create recording audio file", e);
- }
-
- recorder.setOutputFile(audioFile.getAbsolutePath());
-
- try {
- recorder.prepare();
- } catch (IllegalStateException e) {
- throw new RuntimeException("IllegalStateException on MediaRecorder.prepare", e);
- } catch (IOException e) {
- throw new RuntimeException("IOException on MediaRecorder.prepare", e);
- }
- isRecoding=true;
- recorder.start();
- }
-
-
- public void onClick(View v) {
-
- switch (v.getId())
- {
-
- case R.id.StartRecording:
- {
- Toast.makeText(MainActivity.this,"開始錄音",Toast.LENGTH_SHORT).show();
- recod();
- break;
- }
-
-
- case R.id.StopRecording:
- {
- if(isRecoding)
- {
- Toast.makeText(MainActivity.this,"中止錄音",Toast.LENGTH_SHORT).show();
- recorder.stop();
- recorder.release();
- }
- break;
- }
-
- default:
- break;
- }
- }
-
- }
- MediaRecoder的其餘API方法:
- setMaxDuration(int);
- setMaxFileSize(long)
- setAudioChannels(int)
- setAudioSamplingRate(int)
- setAudioEncodingBitRate在(int)
3、使用AudioRecord 進行音頻捕獲。——從某個輸入,直接捕獲音頻數據:PCM格式
這種方法是3種之中最爲靈活的,能讓開發者最大限度的處理採集的音頻,同時它捕獲到的音頻是原始音頻PCM格式的!像作變聲處理的須要就必需要用它收集音頻。在實際開發中,它也是最經常使用來採集音頻的手段。也是本文介紹的重點。
- int audioSource=MediaRecorder.AudioSource.MIC;
- int sampleRateInHz=11025 ;
- int channelConfig=AudioFormat.CHANNEL_CONFIGURATION_MONO;
- int audioFormat=AudioFormat.ENCODING_PCM_16BIT;
- 指定緩衝區大小。調用AudioRecord類的getMinBufferSize方法能夠得到。
- AudioRecord record=new AudioRecord(audioSource,sampleRateInHz,channelConfig,audioFormat,bufferSizeInBytes);
佈局xml文件和MediaRecorder Demo中同樣,2個按鈕而已!就不貼了,下面貼出AudioRecord的Demo的源代碼.
- public class MainActivity extends Activity implements OnClickListener {
- Button startRecordingButton, stopRecordingButton;
- File recordingFile;
- boolean isRecording = false;
- AudioRecord audioRecord=null;
- File parent=null;
- int bufferSize=0;
- int sampleRateInHz = 11025;
- int channelConfig = AudioFormat.CHANNEL_CONFIGURATION_MONO;
- int audioFormat = AudioFormat.ENCODING_PCM_16BIT;
- String TAG="AudioRecord";
-
- @Override
- public void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- requestWindowFeature(Window.FEATURE_NO_TITLE);
- setContentView(R.layout.main);
- init();
- initListener();
- }
-
-
- public void init(){
- startRecordingButton = (Button)findViewById(R.id.StartRecordingButton);
- stopRecordingButton = (Button)findViewById(R.id.StopRecordingButton);
-
- bufferSize = AudioRecord.getMinBufferSize(sampleRateInHz,channelConfig, audioFormat);
- audioRecord = new AudioRecord(MediaRecorder.AudioSource.MIC,sampleRateInHz,channelConfig, audioFormat, bufferSize);
-
- parent = new File(Environment.getExternalStorageDirectory().getAbsolutePath()+ "/AudiioRecordTest");
- if(!parent.exists())
- parent.mkdirs();
- }
-
-
- public void initListener(){
- startRecordingButton.setOnClickListener(this);
- stopRecordingButton.setOnClickListener(this);
- }
-
- public void onClick(View v) {
- switch (v.getId()) {
-
- case R.id.StartRecordingButton:
- {
- record();
- break;
- }
-
-
- case R.id.StopRecordingButton:
- {
- stopRecording();
- break;
- }
- }
-
- }
-
-
- public void record() {
- isRecording = true;
- new Thread(new Runnable() {
- @Override
- public void run() {
- isRecording = true;
-
- recordingFile = new File(parent,"audiotest.pcm");
- if(recordingFile.exists()){
- recordingFile.delete();
- }
-
- try {
- recordingFile.createNewFile();
- }
- catch (IOException e) {
- e.printStackTrace();
- Log.e(TAG,"建立儲存音頻文件出錯");
- }
-
-
- try {
- DataOutputStream dos = new DataOutputStream(new BufferedOutputStream(new FileOutputStream(recordingFile)));
- byte[] buffer = new byte[bufferSize];
- audioRecord.startRecording();
- int r = 0;
- while (isRecording) {
- int bufferReadResult = audioRecord.read(buffer,0,bufferSize);
- for (int i = 0; i < bufferReadResult; i++)
- {
- dos.write(buffer[i]);
- }
- r++;
- }
- audioRecord.stop();
- dos.close();
- } catch (Throwable t) {
- Log.e(TAG, "Recording Failed");
- }
-
- }
- }).start();
-
- }
-
-
- public void stopRecording()
- {
- isRecording = false;
- }
-
-
- }
注意這幾個Demo都要添加如下權限:
- <uses-permission android:name="android.permission.RECORD_AUDIO"/>
- <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
4、備註:
MediaRecorder採集音頻的Demo源碼
AudioRecord採集音頻的Demo源碼
原始音頻PCM播放器 使用方法 : 文件->導入->裸數據 而後根據你錄音時的配置填寫相應參數 就能夠播放了!