我在Linux上使用Java音频时遇到问题。这是Ubuntu 14.04上的OpenJDK 8。以下示例因.wav文件from this link而失败:
import java.net.URL;
import javax.sound.sampled.*;
public class PlaySound {
public void play() throws Exception
{
// List all mixers and default mixer
System.out.println("All mixers:");
for (Mixer.Info m : AudioSystem.getMixerInfo())
{
System.out.println(" " + m);
}
System.out.println("Default mixer: " + AudioSystem.getMixer(null).getMixerInfo());
URL url = getClass().getResource("drop.wav");
Clip clip;
AudioInputStream audioInputStream = AudioSystem.getAudioInputStream(url);
clip = AudioSystem.getClip();
System.out.println("Clip format: " + clip.getFormat());
clip.open(audioInputStream);
clip.start();
do { Thread.sleep(100); } while (clip.isRunning());
}
public static void main(String [] args) throws Exception {
(new PlaySound()).play();
}
}
结果如下:
All mixers:
PulseAudio Mixer, version 0.02
default [default], version 4.4.0-31-generic
Intel [plughw:0,0], version 4.4.0-31-generic
Intel [plughw:0,2], version 4.4.0-31-generic
NVidia [plughw:1,3], version 4.4.0-31-generic
NVidia [plughw:1,7], version 4.4.0-31-generic
NVidia [plughw:1,8], version 4.4.0-31-generic
NVidia [plughw:1,9], version 4.4.0-31-generic
Port Intel [hw:0], version 4.4.0-31-generic
Port NVidia [hw:1], version 4.4.0-31-generic
Default mixer: default [default], version 4.4.0-31-generic
Clip format: PCM_SIGNED unknown sample rate, 16 bit, stereo, 4 bytes/frame, big-endian
Exception in thread "main" java.lang.IllegalArgumentException: Invalid format
at org.classpath.icedtea.pulseaudio.PulseAudioDataLine.createStream(PulseAudioDataLine.java:142)
at org.classpath.icedtea.pulseaudio.PulseAudioDataLine.open(PulseAudioDataLine.java:99)
at org.classpath.icedtea.pulseaudio.PulseAudioDataLine.open(PulseAudioDataLine.java:283)
at org.classpath.icedtea.pulseaudio.PulseAudioClip.open(PulseAudioClip.java:402)
at org.classpath.icedtea.pulseaudio.PulseAudioClip.open(PulseAudioClip.java:453)
at PlaySound.play(PlaySound.java:22)
at PlaySound.main(PlaySound.java:29)
显然问题在于选择了PulseAudio混音器,并且由于某种原因它无法播放.wav文件。
如果我将
AudioSystem.getClip()
调用替换为AudioSystem.getClip(null)
,它将选择默认的混音器,那么它将起作用。如何确定选择了兼容的调音台?
更新:按照@Dave的建议,循环浏览可用的混音器,直到找到具有“兼容”格式的混音器,然后看到以下内容:
目标格式(来自
AudioInputStream.getFormat()
)为:PCM_SIGNED 44100.0 Hz, 16 bit, stereo, 4 bytes/frame, little-endian
我遍历所有混音器,每个混音器的源代码行以及每个源代码行的受支持格式,并获得以下匹配:
Mixer: PulseAudio Mixer, version 0.02
Source line: interface SourceDataLine supporting 42 audio formats, and buffers of 0 to 1000000 bytes
Format matches: PCM_SIGNED unknown sample rate, 16 bit, stereo, 4 bytes/frame, little-endian
我确实有一个匹配项(使用
format.matches()
),但仍然收到“无效格式”异常。也许因为匹配的格式显示为“未知采样率”,然后当我尝试打开剪辑时,它发现它实际上并不支持44100 Hz? 最佳答案
如果您可以在用例中使用 SourceDataLine
而不是 Clip
,那么您应该可以执行以下操作:
URL url = getClass().getResource("drop.wav");
SourceDataLine sourceLine;
AudioInputStream audioInputStream = AudioSystem.getAudioInputStream(url);
sourceLine = AudioSystem.getSourceDataLine(audioInputStream.getFormat());
System.out.println("Source format: " + sourceLine.getFormat());
sourceLine.open(audioInputStream);
sourceLine.start();
do { Thread.sleep(100); } while (sourceLine.isRunning());
(请注意,这还没有经过我的测试。)
如果您打算搜索或循环,则只需要
Clip
。如果确实需要搜索或循环的能力,则一个(丑陋的)方法将首先调用
AudioSystem.getClip(null)
,以确保您选择了默认的Mixer
。正如您所注意到的,AudioSystem.getClip()
的语义不是特别可靠。在try / catch中包装所有调用Clip.open
的尝试。如果打开剪辑不适用于默认的混音器,请遍历除默认设置以外的可用Mixer.Info
对象,并分别调用getClip(mixerInfo)
直到一个可用。另一种方法是循环遍历
Mixer.Info
返回的AudioSystem.getMixerInfo()
对象。为每个调用AudioSystem.getMixer(mixerInfo)
以获取Mixer
实例。循环遍历Line.Info
返回的Mixer.getSourceLineInfo()
对象。对于其中的每一个,如果它是DataLine.Info
的实例,则将其强制转换并调用DataLine.Info.getFormats()
以获取受支持的AudioFormat
对象。将这些与AudioInputStream.getFormat()
返回的内容(使用matches
)进行比较,直到找到兼容的。这些都不是特别优雅。首先是对异常的错误使用。第二个似乎有点令人费解,尽管看起来更正确。