【问题标题】:Play WAV files one after the other in Java在Java中一个接一个地播放WAV文件
【发布时间】:2010-06-22 13:46:04
【问题描述】:

我正在尝试一个接一个地播放几个WAV 文件。我试过这个方法:

for (String file : audioFiles) {
    new AePlayWave(file).start();
}

但这会同时播放它们。所以我需要一个看起来像这样的函数:

public void play(Vector<String> audioFiles);

向量包含文件,例如:"test1.wav","test2.wav"

我已经找了四个多小时了,但我似乎找不到可行的解决方案:(

我还尝试将 WAV 文件连接到一个 AudioInputStream。它没有给出任何编译器错误,但声音完全一团糟。代码:

public static AudioInputStream concat(Vector<String> files) throws UnsupportedAudioFileException, IOException {
    AudioInputStream total = AudioSystem.getAudioInputStream(new File(files.get(0)));

    for (int i = 1; i < files.size(); i++) {
        AudioInputStream clip = AudioSystem.getAudioInputStream(new File(files.get(i)));
        total = new AudioInputStream(new SequenceInputStream(total, clip),
                                     total.getFormat(),
                                     total.getFrameLength() + clip.getFrameLength());
    }
    return total;
}

编辑

即使我尝试将前两个文件放在一起,它也会失败:

public static AudioInputStream concat(Vector<String> files) throws UnsupportedAudioFileException, IOException {
    AudioInputStream clip1 = AudioSystem.getAudioInputStream(new File(files.get(0)));
    AudioInputStream clip2 = AudioSystem.getAudioInputStream(new File(files.get(1)));

    AudioInputStream total = new AudioInputStream(
        new SequenceInputStream(clip1, clip2),
        clip1.getFormat(),
        clip1.getFrameLength() + clip2.getFrameLength());

    return total;
}

【问题讨论】:

  • 你尝试连接文件的方法肯定不会像编码的那样工作,因为你在每次迭代时都会覆盖total
  • 是的,但我用总数和下一个剪辑的组合覆盖了它,还是它仍然有问题? (听起来好像我在覆盖它:))我该如何修复它?

标签: java audio concatenation wav sequential


【解决方案1】:

这段代码有点低级,但可以工作:

    byte[] buffer = new byte[4096];
    for (File file : files) {
        try {
            AudioInputStream is = AudioSystem.getAudioInputStream(file);
            AudioFormat format = is.getFormat();
            SourceDataLine line = AudioSystem.getSourceDataLine(format);
            line.open(format);
            line.start();
            while (is.available() > 0) {
                int len = is.read(buffer);
                line.write(buffer, 0, len);
            }
            line.drain(); //**[DEIT]** wait for the buffer to empty before closing the line
            line.close();
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

基本上,您打开一个 AudioInputStream,读取数据并将其写入 SourceDataLine。 write 方法是阻塞的,所以它会因此播放文件。

您可以尝试将Clip 用于相同目的。

【讨论】:

  • 这行得通,但是如果您想播放文件,则每次在每个文件的末尾都会剪切音频: hello.wav and_welcome.wav to_stackoverflow.wav 您听到:hel 和 welco to stackove any想法如何解决这个问题?
  • @Berty:在 line.close() 之前添加 line.drain()
  • 完美的伎俩,现在听起来很完美:D
【解决方案2】:

最终答案(带漏):

public static void play(ArrayList<String> files){
    byte[] buffer = new byte[4096];
    for (String filePath : files) {
        File file = new File(filePath);
        try {
            AudioInputStream is = AudioSystem.getAudioInputStream(file);
            AudioFormat format = is.getFormat();
            SourceDataLine line = AudioSystem.getSourceDataLine(format);
            line.open(format);
            line.start();
            while (is.available() > 0) {
                int len = is.read(buffer);
                line.write(buffer, 0, len);
            }
            line.drain();
            line.close();
        } catch (Exception ex) {
            ex.printStackTrace();
        }
    }
}

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 2023-04-01
    • 1970-01-01
    • 1970-01-01
    • 2012-06-10
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多