Android音頻可視化開(kāi)發(fā)案例說(shuō)明
Android 調(diào)用自帶的錄制音頻程序
Android中有自帶的音頻錄制程序,我們可以通過(guò)指定一個(gè)Action MediaStore.Audio.Media.RECORD_SOUND_ACTION的Intent來(lái)
啟動(dòng)它就可以了。然后在onActivityResult()方法中,獲取Intent的Data,就是錄制的音頻對(duì)應(yīng)的URI。
java代碼:
package eoe.demo;
import android.app.Activity;
import android.content.Intent;
import android.net.Uri;
import android.os.Bundle;
import android.provider.MediaStore;
import android.view.View;
import android.widget.Toast;
/**
* 被實(shí)例演示如何調(diào)用Android自帶的應(yīng)用來(lái)完成Audio的錄入
* 其實(shí)很簡(jiǎn)單,我們需要指定一個(gè)MediaStore.Audio.Media.RECORD_SOUND_ACTION的Action來(lái)啟動(dòng)就可以
* 返回的Data數(shù)據(jù)就是我們錄制的音頻的URI了
*
* 通過(guò)上面這種方式,靈活性不夠高,我們可以利用MediaRecorder類(lèi)來(lái)實(shí)現(xiàn)自己的音頻錄制程序
* MediaRecorder既可以用來(lái)錄制音頻,也可以用來(lái)錄制視頻
* 創(chuàng)建了一個(gè)MediaRecorder實(shí)例后,需要調(diào)用setAudioSource和setAudioEncoder來(lái)初始化
* 通常情況下,在準(zhǔn)備錄制前,我們還需要調(diào)用setOutputFormat()方法來(lái)決定使用的音頻格式,同時(shí)調(diào)用
* setOutputFile()來(lái)指定存放錄制內(nèi)容的文件
*
* 這幾個(gè)方法的調(diào)用順序是:setAudioSource,setOutputFormat,setAudioEncoder,setOutputFile
*
*
*
* @author Administrator
*
*/
public class AudioRecordDemo extends Activity {
public void onCreate(Bundle savedInstanceState){
super.onCreate(savedInstanceState);
setContentView(R.layout.audio_record);
}
public void onActivityResult(int requestCode, int resultCode, Intent data){
//super.onActivityResult(requestCode, resultCode, data);
//這里我們就可以獲取到剛剛錄制的音頻的Uri,可以進(jìn)行播放等操作,這里顯示返回的Uri
if(resultCode == RESULT_OK){
Uri audioPath = data.getData();
Toast.makeText(this, audioPath.toString(), Toast.LENGTH_LONG).show();
}
}
public void onClick(View v){
int id = v.getId();
switch(id){
case R.id.btn1: //調(diào)用Android自帶的音頻錄制應(yīng)用
Intent intent = new Intent(MediaStore.Audio.Media.RECORD_SOUND_ACTION);
startActivityForResult(intent, 0);
break;
case R.id.btn2:
//通過(guò)MediaRecorder類(lèi)來(lái)實(shí)現(xiàn)自己的音頻錄制程序
Intent intent2 = new Intent();
intent2.setClass(this, MyAudioRecord.class);
startActivityForResult(intent2, 1);
break;
case R.id.btn3:
//通過(guò)AudioRecord類(lèi)實(shí)現(xiàn)自己的音頻錄制程序
Intent intent3 = new Intent();
intent3.setClass(this, MyAudioRecord2.class);
startActivityForResult(intent3, 2);
break;
}
}
}
Android 音頻的介紹
最近移植Android,當(dāng)Android能夠在設(shè)備上面運(yùn)行之后,首先想到的是讓音頻設(shè)備跑起來(lái)。“沒(méi)有聲音,再好的戲也出不來(lái)”。本文簡(jiǎn)單介紹一下Android音頻適配層。
這個(gè)世界音頻設(shè)備千變?nèi)f化,Android也不可能為每種設(shè)備都提供支持。Android定義了一個(gè)框架,這個(gè)框架來(lái)適配底層的音頻設(shè)備。該適配層的定義位于:
Java代碼:
hardware/libhardware_legacy/include/hardware_legacy/AudioHardwareInterface.h
要想視頻底層的音頻設(shè)備必須要繼承該文件中定義的AudioStreamOut,AudioStreamIn,AudioHardwareInterface等類(lèi),并實(shí)現(xiàn)createAudioHardware函數(shù)。
下面我們看一下Android創(chuàng)建音頻設(shè)備的代碼,代碼位于:
Java代碼:
frameworks/base/libs/audioflinger/AudioHardwareInterface.cpp
該文件有如下代碼:
Java代碼:
AudioHardwareInterface* AudioHardwareInterface::create()
{
/*
* FIXME: This code needs to instantiate the correct audio device
* interface. For now - we use compile-time switches.
*/
AudioHardwareInterface* hw = 0;
char value[PROPERTY_VALUE_MAX];
#ifdef GENERIC_AUDIO
hw = new AudioHardwareGeneric();
#elseif (property_get("ro.kernel.qemu", value, 0)) {
LOGD("Running in emulation - using generic audio driver");
hw = new AudioHardwareGeneric();
}
else {
LOGV("Creating Vendor Specific AudioHardware");
hw = createAudioHardware();
}
#endif
if (hw->initCheck() != NO_ERROR) {
LOGW("Using stubbed audio hardware. No sound will be produced.");
delete hw;
hw = new AudioHardwareStub();
}
#ifdef WITH_A2DP
hw = new A2dpAudioInterface(hw);
#endif
#ifdef ENABLE_AUDIO_DUMP
recorded in the file.
LOGV("opening PCM dump interface");
hw = new AudioDumpInterface(hw); // replace interface
#endif
return hw;
}
從代碼中我們可以看出如果定義了GENERIC_AUDIO的宏,則會(huì)創(chuàng)建AudioHardwareGeneric,如果是模擬器的話,AudioHardwareGeneric會(huì)不能初始化,進(jìn)而創(chuàng)建AudioHardwareStub。這兩個(gè)類(lèi)都是Audio設(shè)備的適配層,是Android默認(rèn)提供的。模擬器都是用AudioHardwareStub,不會(huì)有聲音輸出。設(shè)備都是用AudioHardwareGeneric,因?yàn)槟J(rèn)GENERIC_AUDIO是設(shè)置的。
一般我們只關(guān)心AudioHardwareGeneric實(shí)現(xiàn),誰(shuí)會(huì)去給模擬器去調(diào)試聲音呢,反正我沒(méi)這個(gè)閑心。首先說(shuō)明一下這個(gè)音頻適配層是Android自帶的,可以保證你的音頻設(shè)備正常運(yùn)行,但是不能發(fā)揮設(shè)備的最佳性能。通過(guò)后面的描述你將會(huì)了解。AudioHardwareGeneric的定義位于:
Java代碼:
frameworks/base/libs/audioflinger/AudioHardwareGeneric.cpp
上面就是eoe給我們介紹音頻用途,如果有什么不明白的就多看看android的源碼,這樣有助與你對(duì)音頻的理解。
先看一下效果圖:

public class FFTActivity extends Activity implements OnClickListener{
private Button button;
private ImageView imageView;
private int frequency = 8000;
private int channelConfiguration = AudioFormat.CHANNEL_CONFIGURATION_MONO;
private int audioEncoding = AudioFormat.ENCODING_PCM_16BIT;
private RealDoubleFFT transformer;
private int blockSize = 256;
private boolean started = false;
private Canvas canvas;
private Paint paint;
private Bitmap bitmap;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.fft);
button = (Button) findViewById(R.id.fft_button);
button.setOnClickListener(this);
imageView = (ImageView) findViewById(R.id.fft_imageView);
transformer = new RealDoubleFFT(blockSize);
bitmap = Bitmap.createBitmap(256, 100, Bitmap.Config.ARGB_8888);
canvas = new Canvas(bitmap);
paint = new Paint();
paint.setColor(Color.GREEN);
imageView.setImageBitmap(bitmap);
}
private class RecordAudio extends AsyncTask<Void, double[], Void> {
@Override
protected Void doInBackground(Void... params) {
int bufferSize = AudioRecord.getMinBufferSize(frequency,
channelConfiguration, audioEncoding);
AudioRecord audioRecord = new AudioRecord(
MediaRecorder.AudioSource.MIC, frequency,
channelConfiguration, audioEncoding, bufferSize);
short[] buffer = new short[blockSize];
double[] toTransform = new double[blockSize];
audioRecord.startRecording();
while (started) {
//將record的數(shù)據(jù) 讀到buffer中,但是我認(rèn)為叫做write可能會(huì)比較合適些。
int bufferResult = audioRecord.read(buffer, 0, blockSize);
for (int i = 0; i < bufferResult; i++) {
toTransform<i> = (double) buffer<i> / Short.MAX_VALUE;
}
transformer.ft(toTransform);
publishProgress(toTransform);
}
audioRecord.stop();
return null;
}
@Override
protected void onProgressUpdate(double[]... values) {
super.onProgressUpdate(values);
canvas.drawColor(Color.BLACK);
for (int i = 0; i < values[0].length; i++) {
int x=i;
int downy=(int)(100-(values[0]<i>)*10);
int upy=100;
canvas.drawLine(x, downy, x, upy, paint);
}
imageView.invalidate();
}
}
@Override
public void onClick(View v) {
started=true;
new RecordAudio().execute();
}
}
android音頻可視化的原理是使用離散傅里葉變換,但是數(shù)學(xué)不好的同學(xué)不要擔(dān)心,有開(kāi)源的java離散傅里葉變換的代碼?。≈苯拥絯ww.netlib.org/fftpack/jfftpack.tgz,直接將里面javasource目錄拖動(dòng)到(ca目錄)src即可!!
相關(guān)文章
談?wù)凙ndroid的三種網(wǎng)絡(luò)通信方式
Android平臺(tái)有三種網(wǎng)絡(luò)接口可以使用,他們分別是:java.net.*(標(biāo)準(zhǔn)Java接口)、Org.apache接口和Android.net.*(Android網(wǎng)絡(luò)接口)。本文詳細(xì)的介紹,有興趣的可以了解一下。2017-01-01Flutter實(shí)現(xiàn)仿微信分享功能的示例代碼
Flutter 用來(lái)快速開(kāi)發(fā) Android iOS平臺(tái)應(yīng)用,在Flutter 中,通過(guò) fluwx或者fluwx_no_pay 插件可以實(shí)現(xiàn)微信分享功能,本文將具體介紹實(shí)現(xiàn)的示例代碼,需要的可以參考一下2022-01-01Android學(xué)習(xí)筆記之ActionBar Item用法分析
這篇文章主要介紹了Android學(xué)習(xí)筆記之ActionBar Item用法,結(jié)合實(shí)例形式分析了ActionBar Item的具體功能與相關(guān)使用技巧,需要的朋友可以參考下2017-05-05android xml實(shí)現(xiàn)按鈕的圓角、陰影效果及按下變化效果的實(shí)現(xiàn)代碼
這篇文章主要介紹了android xml實(shí)現(xiàn)按鈕的圓角、陰影效果以及按下變化效果,通過(guò)五個(gè)xml文件實(shí)現(xiàn)按鈕的圓角陰影效果,代碼也很簡(jiǎn)單,需要的朋友可以參考下2021-05-05android中intent傳遞list或者對(duì)象的方法
這篇文章主要介紹了android中intent傳遞list或者對(duì)象的方法,分析羅列了常用的幾種方法,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-01-01Android Activity之間的數(shù)據(jù)傳遞方法總結(jié)
這篇文章主要給大家總結(jié)介紹了關(guān)于Android Activity之間的數(shù)據(jù)傳遞方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)各位Android開(kāi)發(fā)者們具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-06-06Android ViewPager實(shí)現(xiàn)選項(xiàng)卡切換
這篇文章主要介紹了Android ViewPager實(shí)現(xiàn)選項(xiàng)卡切換,詳細(xì)分析了ViewPager實(shí)現(xiàn)選項(xiàng)卡切換功能,感興趣的小伙伴們可以參考一下2016-02-02