如何使用JavaScript将音频文件转换为字节数组

3
我正在尝试将一个.wav文件转换为字节数组字符串。我需要在后端完成这个任务,但是定位文件已经成为了一个问题。
        files.forEach(file => {
            let index = files.indexOf(file)
            let reader = new FileReader();
            reader.readAsArrayBuffer(file);
            console.log(reader.result);

            reader.onload = function (event) {
                let byteArray = new Uint8Array(reader.result);
                let FileName = file.name;
                let dataAsByteArrayString = byteArray.toString();
                var listHtml = $list.html();

上述代码使用了npm的fileReader,它需要目标文件。我很难做到这一点,因为这不是文件的前端拖放。 我的生成文件名为"response.wav",如何使用JavaScript和node扩展来转换这个文件呢?谢谢!

转换成“字节数组字符串”? - ccjmne
"DataAsByteArrayString": "82、73、70、70、255、255、255、255、87、65、86、69、102、109、116、32、16、0、0、0、1、0、1、0、34、86、0、0、68、172、0、0、2、0、16、0、76、73、83、84、26、0、0、0、73、78、70、79、73、83、70、84、14、0、0、0、76、97、118、102、53、55、46、55、49、46、49、48、48、0、100、9" - fengelhardt
你是想解析WAV文件,还是只需要通用地读取一个文件? - Brad
2个回答

2
使用包含的 fs 模块,您可以这样读取 wav 文件:
const fs = require('fs');
const path = './path/to/my.wav';

fs.readFile(path, (err, data) => {
    // Data is a Buffer object 

});

关于使用Node.JS Buffer的文档,请参见此处。如果您更感兴趣的是文件转换部分,那么有几个库可供使用。如果您只需要转换功能而不是自己实现,node-fluent-ffmpeg可能适合您。如果您想要自己实现它,那么这个node-wav文件可能是一个很好的参考(太长了,无法在此粘贴)。

如果您需要将Buffer转换为ArrayBuffer,则此SO中提供了一些选项。


谢谢!这帮了很多忙。我已经成功从缓冲区转换为 arrayBuffer 了! - fengelhardt

2
我不确定这是否有帮助,但我最近参与的项目中,我们使用Node缓冲区API解析了一个.wav文件,并使用Node文件API进行了写入。 如果您对代码有更多问题,我可以将您引导到最熟悉该文件的人。希望这能在一定程度上帮助您。 https://github.com/IntelliSound/intelliSound-Server/blob/development/lib/sound-data-parser.js

'use strict';

function ParsedWave(buffer) {
  
  const RIFF_HEADER_OFFSET = 0;
  const FILE_SIZE_OFFSET = 4;
  const RIFF_FORMAT_OFFSET = 8;
  const SUBCHUNK1_ID_OFFSET = 12;
  const AUDIO_FORMAT_OFFSET = 20;
  const NUMBER_OF_CHANNELS_OFFSET = 22;
  const SAMPLE_RATE_OFFSET = 24;
  const BITS_PER_SAMPLE_OFFSET = 34;
  const SUBCHUNK2_ID_OFFSET = 36;
  const SUBCHUNK2_SIZE_OFFSET = 40;
  const DATA_OFFSET = 44;
  
  this.buffer = buffer;
  this.riff = buffer.slice(RIFF_HEADER_OFFSET, RIFF_HEADER_OFFSET + 4).toString('utf8');
  this.fileSize = buffer.readUInt32LE(FILE_SIZE_OFFSET);
  this.riffType = buffer.slice(RIFF_FORMAT_OFFSET, RIFF_FORMAT_OFFSET + 4).toString('utf8');
  this.subChunk1Id = buffer.slice(SUBCHUNK1_ID_OFFSET, SUBCHUNK1_ID_OFFSET + 4).toString('utf8');
  this.audioFormat = buffer.readUInt16LE(AUDIO_FORMAT_OFFSET);
  this.numberOfChannels = buffer.readUInt16LE(NUMBER_OF_CHANNELS_OFFSET);
  this.sampleRate = buffer.readUInt32LE(SAMPLE_RATE_OFFSET);
  this.bitsPerSample = buffer.readUInt16LE(BITS_PER_SAMPLE_OFFSET);
  this.subChunk2Id = buffer.slice(SUBCHUNK2_ID_OFFSET, SUBCHUNK2_ID_OFFSET + 4).toString('utf8');
  this.subChunk2Size = buffer.readUInt32LE(SUBCHUNK2_SIZE_OFFSET);
  this.data = buffer.slice(DATA_OFFSET, this.subChunk2Size + DATA_OFFSET);
}

// Andrew - The bufferMapper function is going to accept a parsed wave-file and output
//          an array of values corresponding to the data subchunk in a format which can
//          be accepted as input to the neural network.

const bufferMapper = parsedWave => {
  
  const SIXTEEN_BIT_ZERO = 32768;
  const SIXTEEN_BIT_MAX = 65535;
  
  parsedWave.neuralArray = [];
  
  for (let i = 0; i < parsedWave.data.length; i += 2) {
    const sample = parsedWave.data.readInt16LE(i);
    const unsignedSample = sample + SIXTEEN_BIT_ZERO;
    const sigmoidSample = unsignedSample / SIXTEEN_BIT_MAX;
    parsedWave.neuralArray.push(sigmoidSample);
  }
  return parsedWave;
};

module.exports = data => {

  const parsedWaveFile = new ParsedWave(data);

  if (parsedWaveFile.riff !== 'RIFF') {
    throw new TypeError('incorrect file type, must be RIFF format');
  }

  if (parsedWaveFile.fileSize > 10000000) {
    throw new TypeError('file too large, please limit file size to less than 10MB');
  }

  if (parsedWaveFile.riffType !== 'WAVE') {
    throw new TypeError('file must be a WAVE');
  }

  if (parsedWaveFile.subChunk1Id !== 'fmt ') {
    throw new TypeError('the first subchunk must be fmt');
  }

  if (parsedWaveFile.audioFormat !== 1) {
    throw new TypeError('wave file must be uncompressed linear PCM');
  }

  if (parsedWaveFile.numberOfChannels > 2) {
    throw new TypeError('wave file must have 2 or less channels');
  }

  if (parsedWaveFile.sampleRate > 48000) {
    throw new TypeError('wave file must have sample rate of less than 48k');
  }

  if (parsedWaveFile.bitsPerSample !== 16) {
    throw new TypeError(`file's bit depth must be 16`);
  }

  if (parsedWaveFile.subChunk2Id !== 'data') {
    throw new TypeError('subchunk 2 must be data');
  }

  const neuralMappedWaveFile = bufferMapper(parsedWaveFile);

  return neuralMappedWaveFile;
};


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接