1. 设置采集音频格式

    1. WAVEFORMATEX waveform; //采集音频的格式,结构体
    2. waveform.wFormatTag = WAVE_FORMAT_PCM;//声音格式为PCM
    3. waveform.nSamplesPerSec = 8000;//采样率,16000次/秒
    4. waveform.wBitsPerSample = 16;//采样比特,16bits/次
    5. waveform.nChannels = 1;//采样声道数,2声道
    6. waveform.nAvgBytesPerSec = 16000;//每秒的数据率,就是每秒能采集多少字节的数据
    7. waveform.nBlockAlign = 2;//一个块的大小,采样bit的字节数乘以声道数
    8. waveform.cbSize = 0;//一般为0
  2. 提前准备好的录音数据缓存
    1. m_pRecoderBuf = new unsigned char[20 * 1024];
    2. m_recoderNum = 20;
    3. m_recoderBufLen = 20 * 1024;
    4. m_pWaveHdr  = new WAVEHDR[20];
    5. memset(m_pRecoderBuf, 0, 20 * 1024);
    6. memset(m_pWaveHdr, 0, sizeof(WAVEHDR) * 20);
  3. 打开录音设备
    1. MMRESULT mmres = waveInOpen(&m_hWaveIn, WAVE_MAPPER, &waveform, (DWORD_PTR)WaveInProc, (DWORD_PTR)this, CALLBACK_FUNCTION );
    2. if(mmres != MMSYSERR_NOERROR)
    3. {
    4. // failed, try again.
    5. return;
    6. }

    WAVE_MAPPER表示系统会自己寻找合适的录音设备。

  4. 准备数据空间
    1. for (int i = 0; i < m_recoderNum; ++i)
    2. {
    3. m_pWaveHdr[i].lpData = (char*)m_pRecoderBuf + i * 1024;
    4. m_pWaveHdr[i].dwBufferLength = 1024;
    5. mmres = waveInPrepareHeader(m_hWaveIn, &m_pWaveHdr[i], sizeof(WAVEHDR));
    6. mmres = waveInAddBuffer(m_hWaveIn, &m_pWaveHdr[i], sizeof(WAVEHDR));
    7. }
  5. 开始录音并打开写文件
    1. mmres = waveInStart(m_hWaveIn);
    2. SYSTEMTIME systemtime;
    3. GetLocalTime(&systemtime);
    4. char filename[256];
    5. sprintf_s(filename, 256, "%d-%d-%d_%d_%d_%d.wav", systemtime.wYear, systemtime.wMonth, systemtime.wDay, systemtime.wHour, systemtime.wMinute, systemtime.wSecond);
    6. m_pWavFile = wav_write_open(filename, 8000, 16, 1);
  6. 要提前定义好数据接收回调函数,因为我们用的是CALLBACK_FUNCTION
    1. static BOOL CALLBACK WaveInProc(HWAVEOUT hwo, UINT uMsg, DWORD dwInstance, DWORD dwParam1, DWORD dwParam2);
    2. BOOL   WaveInProcImpl(HWAVEOUT hwo, UINT uMsg, DWORD dwParam1, DWORD dwParam2);

    函数里的处理是:

    1. BOOL CAudioFileConvertDlg::WaveInProc(HWAVEOUT hwo, UINT uMsg, DWORD dwInstance, DWORD dwParam1, DWORD dwParam2)
    2. {
    3. CAudioFileConvertDlg* pPlayer = (CAudioFileConvertDlg*)dwInstance;
    4. return pPlayer->WaveInProcImpl(hwo, uMsg, dwParam1, dwParam2);
    5. }
    6. BOOL CAudioFileConvertDlg::WaveInProcImpl(HWAVEOUT hwo, UINT uMsg, DWORD dwParam1, DWORD dwParam2)
    7. {
    8. // 忽略打开和关闭设备时的处理
    9. if(uMsg == WIM_DATA)
    10. {
    11. LPWAVEHDR pHdr = (LPWAVEHDR) dwParam1;
    12. MMRESULT mmres = waveInUnprepareHeader (m_hWaveIn, pHdr, sizeof(WAVEHDR));
    13. //处理数据
    14. if (NULL != m_pWavFile)
    15. {
    16. wav_write_data(m_pWavFile, (unsigned char *)(pHdr->lpData), pHdr->dwBytesRecorded);
    17. }
    18. //重新准备数据
    19. mmres = waveInPrepareHeader(m_hWaveIn, pHdr, sizeof(WAVEHDR));
    20. mmres = waveInAddBuffer(m_hWaveIn, pHdr, sizeof(WAVEHDR));
    21. }
    22. return TRUE;
    23. }
  7. 关闭录音并关闭wav写文件
    1. MMRESULT mmres = waveInStop(m_hWaveIn);
    2. mmres = waveInClose(m_hWaveIn);
    3. wav_write_close(m_pWavFile);
    4. if (NULL != m_pRecoderBuf)
    5. {
    6. delete []m_pRecoderBuf;
    7. m_pRecoderBuf = NULL;
    8. }
    9. if (NULL != m_pWaveHdr)
    10. {
    11. delete []m_pWaveHdr;
    12. m_pWaveHdr = NULL;
    13. }
05-11 00:35