原始PCM音頻數(shù)據(jù)轉(zhuǎn)存為WAV文件

1跛十、簡(jiǎn)介

我們獲取到的音頻數(shù)據(jù),可能是需要存儲(chǔ)的秕硝。但是不能只存原始的數(shù)據(jù)芥映,需要一定的格式來(lái)存儲(chǔ)。比如圖片的存儲(chǔ)格式有JPEG远豺、PNG奈偏、GIF等。視頻有mp4躯护、avi惊来、RMVB等。音頻的格式有mp3棺滞、wav裁蚁、WMA等。這些文件都是可以在播放器中直接播放的继准,但是音頻錄制的時(shí)候采樣率枉证、聲道數(shù)等等這些必要的參數(shù)是需要讓播放器知道的,不然播放器就不能正常的播放音頻文件移必。所以在存儲(chǔ)的時(shí)候我們會(huì)為數(shù)據(jù)加上指定的“頭”室谚,這里面包括了音頻的采樣率、聲道等參數(shù)信息。下面我們就學(xué)習(xí)wav格式舞萄。

2眨补、wav格式的文件頭

通過(guò)參考這個(gè)網(wǎng)站:http://soundfile.sapp.org/doc/WaveFormat

image.png

文件頭包括三個(gè)部分

  • 第一部分通過(guò)“ChunkID”來(lái)表示這是一個(gè) “RIFF”格式的文件管削,通過(guò)“Format”填入“WAVE”來(lái)標(biāo)識(shí)這是一個(gè) wav 文件倒脓。而“ChunkSize”則記錄了整個(gè) wav 文件的字節(jié)數(shù)。
  • 第二部分屬于“fmt”信息塊含思,主要記錄了本 wav 音頻文件的詳細(xì)音頻參數(shù)信息崎弃,例如:通道數(shù)、采樣率含潘、位寬等等饲做。
  • 第三部分屬于“data”信息塊,由“Subchunk2Size”這個(gè)字段來(lái)記錄后面存儲(chǔ)的二進(jìn)制原始音頻數(shù)據(jù)的長(zhǎng)度遏弱。

第一部分和第二部分義工占36個(gè)字節(jié)盆均、第三部分Subchunk2ID、和Subchunk2Size各占4字節(jié)漱逸,這44字節(jié)是文件頭固定的長(zhǎng)度泪姨。后面的字節(jié)就是真正的數(shù)據(jù)部分。所以當(dāng)我們拿到了原始PCM數(shù)據(jù)后饰抒,加入前44字節(jié)文件頭肮砾,保存為wav格式。這樣就可以在其他播放器上播放了袋坑。

3仗处、工具代碼

/**
 * 原始PCM數(shù)據(jù)轉(zhuǎn)WAV
 */
public class PcmToWavUtil {

    /**
     * 采樣率
     */
    private int mSampleRate;
    /**
     * 聲道數(shù)
     */
    private int mChannel;


    /**
     * @param sampleRate sample rate、采樣率
     * @param channel    channel枣宫、聲道
     */
    PcmToWavUtil(int sampleRate, int channel) {
        this.mSampleRate = sampleRate;
        this.mChannel = channel;
    }


    /**
     * pcm文件轉(zhuǎn)wav文件
     */
    public ByteArrayOutputStream pcmToWav(ByteArrayOutputStream pcmBaos) {

        //音頻數(shù)據(jù)的長(zhǎng)度
        long totalAudioLen;
        //音頻數(shù)據(jù)的長(zhǎng)度和文件頭中36字節(jié)的總和
        long totalDataLen;
        long longSampleRate = mSampleRate;
        int channels = mChannel == AudioFormat.CHANNEL_IN_MONO ? 1 : 2;
        long byteRate = 16 * mSampleRate * channels / 8;
        ByteArrayOutputStream wavBaos = null;
        try {
            totalAudioLen = pcmBaos.size();
            //由于文件頭中婆誓,后8個(gè)字節(jié)也是屬于數(shù)據(jù)部分,所以這里只加上前面的36字節(jié)
            totalDataLen = totalAudioLen + 36;

            wavBaos = new ByteArrayOutputStream();

            //添加頭
            writeWaveFileHeader(wavBaos, totalAudioLen, totalDataLen,
                    longSampleRate, channels, byteRate);

            wavBaos.write(pcmBaos.toByteArray());
            wavBaos.flush();
            return wavBaos;
        } catch (IOException e) {
            try {
                pcmBaos.close();
            } catch (IOException e1) {
                e1.printStackTrace();
            }
            e.printStackTrace();
        } finally {
            if (pcmBaos != null) {
                try {
                    pcmBaos.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
            if (wavBaos != null) {
                try {
                    wavBaos.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
        return null;
    }

    /**
     * 加入wav文件頭
     */
    private void writeWaveFileHeader(ByteArrayOutputStream wavBaos, long totalAudioLen,
                                     long totalDataLen, long longSampleRate, int channels, long byteRate)
            throws IOException {
        byte[] header = new byte[44];
        // RIFF/WAVE header
        header[0] = 'R';
        header[1] = 'I';
        header[2] = 'F';
        header[3] = 'F';
        header[4] = (byte) (totalDataLen & 0xff);
        header[5] = (byte) ((totalDataLen >> 8) & 0xff);
        header[6] = (byte) ((totalDataLen >> 16) & 0xff);
        header[7] = (byte) ((totalDataLen >> 24) & 0xff);
        //Format 'WAVE'
        header[8] = 'W';
        header[9] = 'A';
        header[10] = 'V';
        header[11] = 'E';
        //Subchunk1ID 'fmt'
        header[12] = 'f';
        header[13] = 'm';
        header[14] = 't';
        header[15] = ' ';
        // 4 bytes: size of 'fmt ' chunk
        header[16] = 16;
        header[17] = 0;
        header[18] = 0;
        header[19] = 0;
        // format = 1
        header[20] = 1;
        header[21] = 0;
        //通道數(shù)
        header[22] = (byte) channels;
        header[23] = 0;
        //采樣率
        header[24] = (byte) (longSampleRate & 0xff);
        header[25] = (byte) ((longSampleRate >> 8) & 0xff);
        header[26] = (byte) ((longSampleRate >> 16) & 0xff);
        header[27] = (byte) ((longSampleRate >> 24) & 0xff);
        //音頻數(shù)據(jù)攢送率
        header[28] = (byte) (byteRate & 0xff);
        header[29] = (byte) ((byteRate >> 8) & 0xff);
        header[30] = (byte) ((byteRate >> 16) & 0xff);
        header[31] = (byte) ((byteRate >> 24) & 0xff);
        // block align
        header[32] = (byte) (channels * 16 / 8);
        header[33] = 0;
        // 每個(gè)樣本的數(shù)據(jù)位數(shù)
        header[34] = 16;
        header[35] = 0;
        //data
        header[36] = 'd';
        header[37] = 'a';
        header[38] = 't';
        header[39] = 'a';
        header[40] = (byte) (totalAudioLen & 0xff);
        header[41] = (byte) ((totalAudioLen >> 8) & 0xff);
        header[42] = (byte) ((totalAudioLen >> 16) & 0xff);
        header[43] = (byte) ((totalAudioLen >> 24) & 0xff);
        wavBaos.write(header);
    }
}

上面的writeWaveFileHeader()方法就可以很明確的看到前44字節(jié)的創(chuàng)建過(guò)程也颤。
如果我們需要讀取wav文件進(jìn)行播放洋幻,就可以拿出前44字節(jié),然后讀出需要的參數(shù)使用AudioTrack進(jìn)行播放歇拆。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末鞋屈,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子故觅,更是在濱河造成了極大的恐慌厂庇,老刑警劉巖,帶你破解...
    沈念sama閱讀 218,525評(píng)論 6 507
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件输吏,死亡現(xiàn)場(chǎng)離奇詭異权旷,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,203評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門拄氯,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)躲查,“玉大人,你說(shuō)我怎么就攤上這事译柏×椭螅” “怎么了?”我有些...
    開封第一講書人閱讀 164,862評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵鄙麦,是天一觀的道長(zhǎng)典唇。 經(jīng)常有香客問(wèn)我,道長(zhǎng)胯府,這世上最難降的妖魔是什么介衔? 我笑而不...
    開封第一講書人閱讀 58,728評(píng)論 1 294
  • 正文 為了忘掉前任,我火速辦了婚禮骂因,結(jié)果婚禮上炎咖,老公的妹妹穿的比我還像新娘。我一直安慰自己寒波,他們只是感情好乘盼,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,743評(píng)論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著影所,像睡著了一般个束。 火紅的嫁衣襯著肌膚如雪钧汹。 梳的紋絲不亂的頭發(fā)上狸窘,一...
    開封第一講書人閱讀 51,590評(píng)論 1 305
  • 那天饿肺,我揣著相機(jī)與錄音,去河邊找鬼卷中。 笑死矛双,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的蟆豫。 我是一名探鬼主播议忽,決...
    沈念sama閱讀 40,330評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼十减!你這毒婦竟也來(lái)了栈幸?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,244評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤帮辟,失蹤者是張志新(化名)和其女友劉穎速址,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體由驹,經(jīng)...
    沈念sama閱讀 45,693評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡芍锚,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,885評(píng)論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片并炮。...
    茶點(diǎn)故事閱讀 40,001評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡默刚,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出逃魄,到底是詐尸還是另有隱情荤西,我是刑警寧澤,帶...
    沈念sama閱讀 35,723評(píng)論 5 346
  • 正文 年R本政府宣布嗅钻,位于F島的核電站皂冰,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏养篓。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,343評(píng)論 3 330
  • 文/蒙蒙 一赂蕴、第九天 我趴在偏房一處隱蔽的房頂上張望柳弄。 院中可真熱鬧,春花似錦概说、人聲如沸碧注。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,919評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)萍丐。三九已至,卻和暖如春放典,著一層夾襖步出監(jiān)牢的瞬間逝变,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,042評(píng)論 1 270
  • 我被黑心中介騙來(lái)泰國(guó)打工奋构, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留壳影,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,191評(píng)論 3 370
  • 正文 我出身青樓弥臼,卻偏偏與公主長(zhǎng)得像宴咧,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子径缅,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,955評(píng)論 2 355

推薦閱讀更多精彩內(nèi)容