FFmpeg PCM轉WAV

一、WAVE 文件格式介紹

WAVE 文件是基于 Microsoft RIFF 標準的文件格式。RIFF 格式文件以文件頭開始,后面跟一系列聲音數據。WAVE 文件通常只是具有單個 RIFF chunk 的文件,該 RIFF chunk 由 fmt chunk 和 data chunk 兩個 sub chunk 組成,其中 fmt chunk 包含音頻數據的采樣率,位深,聲道數等信息,data chunk 包含真正的音頻數據。WAV文件頭一般是 44 字節(jié)。WAV 文件標準格式如下:

我們用一個數據格式使用十六進制展示的大小為 72 字節(jié) WAVE 格式文件舉例:

WAV 文件格式文檔:http://soundfile.sapp.org/doc/WaveFormat/
參考文檔:https://wavefilegem.com/how_wave_files_work.html

二、使用 FFmpeg 命令行的方式

ffmpeg -ar 44100 -ac 2 -f s16le -i test.pcm test.wav

使用上面命令生成的 WAV 文件頭信息有 78 個字節(jié),對比 44 字節(jié)頭文件,增加了一個 34 字節(jié)大小的 LIST chunk。想去掉 LIST chunk 可以加上一個輸出文件參數 -bitexact 如下:

ffmpeg -ar 44100 -ac 2 -f s16le -i test.pcm -bitexact test.wav

二、使用 FFmpeg API 的方式

1、創(chuàng)建 WAVE 頭信息結構體

// WAV 文件格式頭信息(44字節(jié))
typedef struct WAVHeader {
    // RIFF Chunk ID
    char riffCkID[4] = {'R', 'I', 'F', 'F'};
    // RIFF Chunk Size
    int32_t riffCkSize;
    // Format
    char format[4] = {'W', 'A', 'V', 'E'};
    // Format Chunk ID
    char fmtCkID[4] = {'f', 'm', 't', ' '};
    // Format Chunk Size
    int32_t fmtCkSize = 16;
    // 音頻格式 1=PCM 3=Floating Point
    int16_t audioFormat = 1;
    // 聲道數
    int16_t channels;
    // 采樣率
    int32_t sampleRate;
    // 字節(jié)率
    int32_t byteRate = sampleRate * blockAlign;
    int16_t blockAlign = (channels * bitsPerSample) >> 3;
    // 位深
    int16_t bitsPerSample;
    // Data Chunk ID
    char dataCkID[4] = {'d', 'a', 't', 'a'};
    // Data Chunk Size
    int32_t dataCkSize;
} WAVHeader;

2、 PCM 數據文件轉成 WAV 文件主要代碼

QFile pcmFile(PCM_FILE_NAME);
// 獲取 PCM 文件大小
int pcmDataSize = pcmFile.size();
WAVHeader wavHeader;
// 設置采樣率
wavHeader.sampleRate = 44100;
// 設置位深
wavHeader.bitsPerSample = 16;
// 設置聲道數
wavHeader.channels = 2;
// 設置 data chunk size,即實際 PCM 數據的長度,單位字節(jié)
wavHeader.dataCkSize = pcmDataSize;
// 設置 RIFF chunk size,RIFF chunk size 不包含 RIFF Chunk ID 和 RIFF Chunk Size的大小,所以用 PCM 數據大小加 RIFF 頭信息大小減去 RIFF Chunk ID 和 RIFF Chunk Size的大小
wavHeader.riffCkSize = (pcmDataSize + sizeof (WAVHeader) - 4 - 4);
// 調用封裝的方法,FFmpegUtils 是自定義的一個類,傳入頭信息,輸入的 PCM 文件路徑,輸出的 WAV 文件路徑
FFmpegUtils::pcm2wav(wavHeader, PCM_FILE_NAME, WAV_FILE_NAME);
void FFmpegUtils::pcm2wav(WAVHeader &header, const char *pcmFilename, const char *wavFilename)
{
    QFile pcmFile(pcmFilename);
    if (!pcmFile.open(QFile::ReadOnly)) {
        qDebug() << “打開 PCM 文件失敗: " << pcmFilename;
        return;
    }

    QFile wavFile(wavFilename);
    if (!wavFile.open(QFile::WriteOnly)) {
        qDebug() << “打開 WAV 文件失敗: " << wavFilename;
        return;
    }

    wavFile.write((const char *) &header, sizeof (WAVHeader));

    int size = 0;
    char buffer[BUFFER_SIZE];
    while ((size = pcmFile.read(buffer, BUFFER_SIZE)) > 0) {
        wavFile.write(buffer, size);
    }

    pcmFile.close();
    wavFile.close();
}
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
【社區(qū)內容提示】社區(qū)部分內容疑似由AI輔助生成,瀏覽時請結合常識與多方信息審慎甄別。
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發(fā)布,文章內容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務。

相關閱讀更多精彩內容

友情鏈接更多精彩內容