是否可以生成长度小于1秒的音调/音频?或立即切断玩家?

时间:2012-09-26 14:25:58

标签: android audio audiotrack

我正在尝试创建一个应用程序,根据您在屏幕上的触摸位置生成音调(当前频率由触摸点的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);

}

1 个答案:

答案 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将具有所需的缓冲区大小。