0

我有一个可以正常工作的软合成器,它可以正确输出和播放样本,但是我有一个巨大的延迟,大约一秒。我的代码基于此处找到的一篇文章:http ://www.drdobbs.com/jvm/creating-music-components-in-java/229700113?pgno=2

我忽略了什么吗?样本的生成不是问题,它发生得很快而且很简单。

我尝试将缓冲区大小更改为几个不同的值,但没有成功。我目前正在 OSX 机器上进行测试,这可能是问题吗?

仅供参考,完成绝不是错误的。到了沉默的时候,我只需将 0 的样本输入缓冲区。

public class Player extends Thread {
    public static final int SAMPLE_RATE = 44100;
    public static final int BUFFER_SIZE = 2200;
    public static final int SAMPLES_PER_BUFFER = BUFFER_SIZE / 2;
    private static final int SAMPLE_SIZE = 16; // Don't change
    private static final int CHANNELS = 1;
    private static final boolean SIGNED = true;
    private static final boolean BIG_ENDIAN = true;
    private AudioFormat format;
    private DataLine.Info info;
    private SourceDataLine audioLine;
    private boolean done;
    private byte[] sampleData = new byte[BUFFER_SIZE];
    private Oscillator osc;

    public Player(Oscillator osc) {
        format = new AudioFormat(SAMPLE_RATE, SAMPLE_SIZE, CHANNELS, SIGNED, BIG_ENDIAN);
        info = new DataLine.Info(SourceDataLine.class, format);
        this.osc = osc;
    }

    public void run() {
        done = false;
        int bytesRead = 0;

        try {
            audioLine = (SourceDataLine) AudioSystem.getLine(info);
            audioLine.open(format);
            audioLine.start();

            while ((bytesRead != -1) && !done) {
                bytesRead = osc.getSamples(sampleData);

                if (bytesRead > 0) {
                    audioLine.write(sampleData, 0, bytesRead);
                }
            }
        } catch (LineUnavailableException e) {
            e.printStackTrace();  //To change body of catch statement use File | Settings | File Templates.
        } finally {
            audioLine.drain();
            audioLine.close();
        }
    }
}
4

1 回答 1

0

您似乎没有设置音频系统使用的音频缓冲区大小 - 而是SAMPLES_PER_BUFFER用于控制一次生成的样本数量。

这些不是一回事——操作系统将请求渲染它使用的任何缓冲区大小(在 MacOSX 上,样本是通过回调提取的)。

这段代码:

        while ((bytesRead != -1) && !done) {
            bytesRead = osc.getSamples(sampleData);

            if (bytesRead > 0) {
                audioLine.write(sampleData, 0, bytesRead);

只会填满音频线的缓冲区,直到它阻塞。渲染请求将到达,清空缓冲区,然后再次解除阻塞。这将以音频缓冲区大小的速率发生。

于 2013-06-05T08:31:30.780 回答