我正在尝试创建一个应用程序,根据您在屏幕上的触摸位置生成音调(当前频率由触摸点的y值决定)但是我无法生成更少的音调没有崩溃我的应用程序超过一秒!我从the answer here获取代码并在触摸屏幕时将其调整为播放,但是当我将持续时间变量更改为小于1的值时,应用程序崩溃了!是否有可能产生小于1秒的音调?
我想要创建不到一秒的音调的唯一原因是,当我暂停或停止录音时,音频播放器似乎没有立即切断。这些是我演奏和产生音调的方法:
RelativeLayout background;
SoundPool soundPool;
TextView textView;
private final float duration = 1f; // seconds
private final int sampleRate = 8000;//was 8000
private final int numSamples = (int) duration * sampleRate;
private final double sample[] = new double[numSamples];
private final double freqOfTone = 250; // hz
AudioTrack audioTrack ;
int dispWidth = 0, dispHeight = 0;
Handler handler = new Handler();
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_audio);
background = (RelativeLayout) findViewById(R.id.background);
textView = (TextView) findViewById(R.id.textView2);
soundPool = new SoundPool(1, AudioManager.STREAM_MUSIC, 0);
dispWidth = AppHelper.getDisplayWidth(getApplicationContext());
dispHeight = AppHelper.getDisplayHeight(getApplicationContext());
audioTrack = new AudioTrack(AudioManager.STREAM_MUSIC,
sampleRate, AudioFormat.CHANNEL_OUT_MONO,
AudioFormat.ENCODING_PCM_16BIT, numSamples,
AudioTrack.MODE_STREAM);
final byte genTone[]= genTone(freqOfTone);
background.setClickable(true);
background.setOnTouchListener(new OnTouchListener() {
@Override
public boolean onTouch(View v, MotionEvent event) {
if(event.getAction()==MotionEvent.ACTION_UP)
{
Log.d(LOG_TAG, "Touch up! Stopping tone...");
audioTrack.pause();
audioTrack.flush();
}
else
{
//these apparently cause immediate stop
audioTrack.pause();
audioTrack.flush();
int R = (int) (event.getY()%255);
int G = (int)(event.getX()%255);
int B = (int) ((event.getY()+event.getX())%255);
Log.i(LOG_TAG, "Changine colour to "+R+","+G+","+B);
background.setBackgroundColor(Color.rgb(R, G, B));
int colorFromPressure = (int) (event.getPressure()*255);
textView.setTextColor(Color.rgb(colorFromPressure,colorFromPressure,colorFromPressure));
playSoundAtFrequency(250+event.getY(), event.getX()/dispWidth);
}
return false;
}
});
}
byte[] genTone(double toneFrequency){
final byte generatedSnd[] = new byte[2 * numSamples];
// fill out the array
for (int i = 0; i < numSamples; ++i) {
sample[i] = Math.sin(2 * Math.PI * i / (sampleRate/toneFrequency));
}
// convert to 16 bit pcm sound array
// assumes the sample buffer is normalised.
int idx = 0;
for (double dVal : sample) {
short val = (short) (dVal * 32767);
generatedSnd[idx++] = (byte) (val & 0x00ff);
generatedSnd[idx++] = (byte) ((val & 0xff00) >>> 8);
}
return generatedSnd;
}
void playSound(final byte[] generatedSnd, float volume){
//these apparently cause immediate stop
audioTrack.pause();
audioTrack.flush();
audioTrack.write(generatedSnd, 0, numSamples);
audioTrack.setStereoVolume(volume, volume);
audioTrack.play();
Log.d(LOG_TAG, "Playing tone...");
}
public void playSoundAtFrequency(final double toneFrequency,float volume)
{
final byte genTone[]= genTone(toneFrequency);
playSound(genTone, volume);
}
答案 0 :(得分:1)
我认为您的崩溃来自于使用的缓冲区大小对于您尝试创建的AudioTrack来说太小了。
int audioBufferSize = AudioTract.getMinBufferSize(sampleRate,
AudioFormat.CHANNEL_OUT_MONO,
AudioFormat.ENCODING_PCM-16BIT);
audioTrack = new AudioTrack(AudioManager.STREAM_MUSIC,
sampleRate, AudioFormat.CHANNEL_OUT_MONO,
AudioFormat.ENCODING_PCM_16BIT, audioBufferSize,
AudioTrack.MODE_STREAM);
然后numSamples仍适用于您的生成缓冲区,AudioTrack将具有所需的缓冲区大小。