最近用libmad做了些mp3解碼的工作,順便也研究了一下windows下播放PCM音頻數據的雙緩沖用法。 libmad的調用在此暫略過不表。 libmad解碼出來的是16bit的PCM數據,調用windows API可對其實現播放。不過如果解碼一段播放一段,聽起來會有一頓一頓的感覺,不流暢,究其原因,是沒有使用雙緩沖。 吭哧吭哧研究了半天,終于編碼實現,播放出來的效果倒也很流暢。 流程如下: 1)聲明兩個WAVEHDR結構waveHeader1,waveHeader2,并分別對其lpData參數分配緩沖buf1,buf2; 2)聲明WAVEFORMATEX結構waveFormat,以及HWAVEOUT結構hWaveOut。調用函數 waveOutOpen( &hWaveOut, WAVE_MAPPER, &waveFormat, (DWORD)waveOutProc, NULL, CALLBACK_FUNCTION ); waveOutProc為回調函數,在后面會提到。 也說一下waveFormat的各參數。根據MSDN解釋,nChannels為通道數,nSamplesPerSec為采樣率,wFormatTag的值為WAVE_FORMAT_PCM,wBitsPerSample為16,nBlockAlign為 nChannels*wBitsPerSample/8,nAvgBytesPerSec為nSamplesPerSec*nBlockAlign; 3)讀入buf1,buf2,并設置好相應長度; 4)將waveHeader1,waveHeader2寫入wave設備: waveOutPrepareHeader( hWaveOut, &waveHeader1, sizeof(WAVEHDR)); waveOutPrepareHeader( hWaveOut, &waveHeader2, sizeof(WAVEHDR)); waveOutWrite( hWaveOut, &waveHeader1, sizeof(WAVEHDR) ); waveOutWrite( hWaveOut, &waveHeader2, sizeof(WAVEHDR) ); 5)關于回調 void CALLBACK waveOutProc( HWAVEOUT hwo, UINT uMsg, DWORD dwInstance, DWORD dwParam1, DWORD dwParam2 ) { if(uMsg == WOM_DONE) { LPWAVEHDR pWaveHeader = (LPWAVEHDR)dwParam1;//系統自動識別是哪一個WAVEHDR播放完畢 waveOutUnprepareHeader( hwo, pWaveHeader, sizeof(WAVEHDR) );//播放完后須調用此函數 //此處填充WAVEHDR的lpdate緩沖 waveOutPrepareHeader( hwo, pWaveHeader, sizeof(WAVEHDR)); waveOutWrite( hwo, pWaveHeader, sizeof(WAVEHDR) ); //... } return ; } 6)播放完畢后,調用waveOutClose,釋放緩沖。其他的一些waveOut函數,如waveOutPause、 waveOutReset等等,在做播放器的時候會用得到。若播放過程中終止,須先調用waveOutReset,再調用waveOutClose。 |