在Java中将音频流转换为WAV字节数组而不使用临时文件

在Java中将音频流转换为WAV字节数组而不使用临时文件,java,audio,wav,javasound,Java,Audio,Wav,Javasound,给定一个名为in的InputStream,其中包含压缩格式的音频数据(如MP3或OGG),我希望创建一个byte数组,其中包含输入数据的WAV转换。不幸的是,如果您尝试这样做,JavaSound会给您带来以下错误: java.io.IOException: stream length not specified 我将wav写入一个临时文件,然后将其读回,从而使其正常工作,如下所示: AudioInputStream source = AudioSystem.getAudioInputStrea

给定一个名为
in
InputStream
,其中包含压缩格式的音频数据(如MP3或OGG),我希望创建一个
byte
数组,其中包含输入数据的WAV转换。不幸的是,如果您尝试这样做,JavaSound会给您带来以下错误:

java.io.IOException: stream length not specified
我将wav写入一个临时文件,然后将其读回,从而使其正常工作,如下所示:

AudioInputStream source = AudioSystem.getAudioInputStream(new BufferedInputStream(in, 1024));
AudioInputStream pcm = AudioSystem.getAudioInputStream(AudioFormat.Encoding.PCM_SIGNED, source);
AudioInputStream ulaw = AudioSystem.getAudioInputStream(AudioFormat.Encoding.ULAW, pcm);
File tempFile = File.createTempFile("wav", "tmp");
AudioSystem.write(ulaw, AudioFileFormat.Type.WAVE, tempFile);
// The fileToByteArray() method reads the file
// into a byte array; omitted for brevity
byte[] bytes = fileToByteArray(tempFile);
tempFile.delete();
return bytes;

这显然不太可取。有更好的方法吗?

问题是大多数音频文件编写器在写入输出流时需要提前知道文件大小。因为你不能提供这个,它总是失败。不幸的是,默认的JavaSoundAPI实现没有任何替代方案

但是您可以尝试使用Tritonus插件中的AudioOutputStream架构(Tritonus是Java声音API的开源实现):

这非常简单

File f = new File(exportFileName+".tmp");
File f2 = new File(exportFileName);
long l = f.length();
FileInputStream fi = new FileInputStream(f);
AudioInputStream ai = new AudioInputStream(fi,mainFormat,l/4);
AudioSystem.write(ai, Type.WAVE, f2);
fi.close();
f.delete();

.tmp文件是一个原始音频文件,结果是一个带头的WAV文件。

我注意到很久以前就有人问过这个问题。如果有任何新人(使用Java 7及以上版本)发现此线程,请注意,有一种更好的新方法可以通过Files.readAllBytes API实现。见:
太晚了,我知道,但我需要这个,所以这是我在这个话题上的两分钱

public void UploadFiles(String fileName, byte[] bFile)
{
    String uploadedFileLocation = "c:\\";

    AudioInputStream source;
    AudioInputStream pcm;
    InputStream b_in = new ByteArrayInputStream(bFile);
    source = AudioSystem.getAudioInputStream(new BufferedInputStream(b_in));
    pcm = AudioSystem.getAudioInputStream(AudioFormat.Encoding.PCM_SIGNED, source);
    File newFile = new File(uploadedFileLocation + fileName);
    AudioSystem.write(pcm, Type.WAVE, newFile);

    source.close();
    pcm.close();
}

这个问题很容易解决,如果你准备了一个能为你创建正确标题的类。在我的示例中,数据进入某个缓冲区,然后我创建了头文件,并在缓冲区中有wav文件。不需要额外的图书馆。只需从我的示例中复制代码

示例如何使用在缓冲区数组中创建正确标头的类:

public void run() {    
    try {    
        writer = new NewWaveWriter(44100);  

        byte[]buffer = new byte[256];  
        int res = 0;  
        while((res = m_audioInputStream.read(buffer)) > 0) {  
            writer.write(buffer, 0, res);  
        }  
    } catch (IOException e) {  
        System.out.println("Error: " + e.getMessage());  
    }    
}    

public byte[]getResult() throws IOException {  
    return writer.getByteBuffer();  
}  

你可以在我的链接下找到NewWaveWriter类。

我必须试一试。我可能还没来得及试试,所以我现在不能接受答案。不过,我会投赞成票。问题是,如果没有临时文件,是否可以这样做。