Java AudioSystem 和 TargetDataLine

Posted

技术标签:

【中文标题】Java AudioSystem 和 TargetDataLine【英文标题】:Java AudioSystem and TargetDataLine 【发布时间】:2009-02-28 17:24:21 【问题描述】:

我正在尝试从我的 PC 的线路输入中捕获音频,为此我正在使用 Audiosystem 类。静态 AudioSystem.write 方法有两种选择之一:写入文件或写入流。我可以让它很好地写入文件,但是每当我尝试写入流时,我都会抛出 java.io.IOException (未指定流长度)。至于我的缓冲区,我使用的是 ByteArrayOutputStream。我应该在其他地方使用或搞乱另一种流吗?

另外在相关主题中,可以通过调用read直接采样(TargetDataLine)中的音频线。这是进行音频捕获还是使用 AudioSystem 的首选方式?

更新 请求的源代码:

final private TargetDataLine line;
final private AudioFormat format;
final private AudioFileFormat.Type fileType;
final private AudioInputStream audioInputStream;
final private ByteArrayOutputStream bos;

// Constructor, etc.

public void run()

    System.out.println("AudioWorker Started");
    try
    
        line.open(format);
        line.start();

        // This commented part is regarding the second part
        // of my question
        // byte[] buff = new byte[512];
        // int bytes = line.read(buff, 0, buff.length);

        AudioSystem.write(audioInputStream, fileType, bos);

    
    catch ( Exception e )
    
        e.printStackTrace();
    

    System.out.println("AudioWorker Finished");



// Stack trace in console
AudioWorker Started
java.io.IOException: stream length not specified
    at com.sun.media.sound.WaveFileWriter.write(Unknown Source)
    at javax.sound.sampled.AudioSystem.write(Unknown Source)
    at AudioWorker.run(AudioWorker.java:41)
AudioWorker Finished

【问题讨论】:

将堆栈跟踪(至少前几帧)发布到您的应用。 【参考方案1】:

来自 AudioSystem.write JavaDoc:

将表示指定文件类型的音频文件的字节流写入提供的输出流。有些文件类型要求将长度写入文件头;除非事先知道长度,否则此类文件不能从头到尾写入。如果音频文件类型的长度为 AudioSystem.NOT_SPECIFIED,则尝试写入此类文件将失败并返回 IOException。

【讨论】:

这听起来像是写入文件(虽然文件也是流..)?但这如何适用于写入输出流?这是否意味着不可能? @srand,我认为这是有道理的,因为你不是流式传输(需要不同的机制),所以基本上你需要在写入输出流之前指定长度。【参考方案2】:

由于 Wave 格式需要在文件开头写入长度,因此作者正在查询您的 AudioInputStreamgetFrameLength 方法。当它返回 NOT_SPECIFIED 时——因为你正在录制尚未指定长度的“实时”数据——作者抛出异常。

面向File 的解决方法是将虚拟数据写入长度字段,然后在写入完成后重新打开文件并覆盖文件的该区域。

使用不需要提前长度的输出格式 (au),或使用返回有效帧长度的 AudioInputStream,或使用 API 的 File 版本。

【讨论】:

据我了解,AudioSystem.write 只写入文件? 那么签名与OutputStream有什么用处呢? @AlessandroDionisi 它适用于某些格式,例如 .au【参考方案3】:

您应该查看Richard Baldwin's Java 声音教程。文章底部有一个完整的源列表,他使用 TargetDataLine 的读取来捕获音频。

【讨论】:

【参考方案4】:

您也可以尝试使用 JMF,它有点麻烦,但比 javax.sound.sampled 的东西好一点。 JMF 页面上有很多教程介绍了如何从线路输入或麦克风通道进行录制。

【讨论】:

以上是关于Java AudioSystem 和 TargetDataLine的主要内容,如果未能解决你的问题,请参考以下文章

AudioSystem.getMixerInfo() 在 Java/gentoo 中一无所获

Java 重复剪辑

使用 Tritonus AudioSystem:将 PCM byte[] 数组转换为 mp3 byte[] 数组

java声音淡出

<一>Android Audio音频框架

Android - 如何获取 AudioManager / AudioSystem 参数列表