VC5.0下实现DirectSound

发表于:2007-07-04来源:作者:点击数: 标签:
武汉石化设计院 周红汉 音乐可以使用户心情愉快,在合适的场合播放恰当的音乐能够使 程序员 和他的VC ++程序焕发出光彩。 API 提供了三种方法来播放WAV 文件: *PlaySound()函数。它可以通过单行编码来播放Wave 格式的声音。此函数有两个限制:必须将声
武汉石化设计院 周红汉

  音乐可以使用户心情愉快,在合适的场合播放恰当的音乐能够使程序员和他的VC ++程序焕发出光彩。
  API 提供了三种方法来播放WAV 文件:
  *PlaySound()函数。它可以通过单行编码来播放Wave 格式的声音。此函数有两个限制:必须将声音数据完整地载入物理内存;数据格式必须被所配置的某一音频驱动器支持。根据经验,PlaySound()适用于100K 以下的文件。
  *MCI(The Media Control Interface)。MCI 功能强大,不仅可以实现WAV 文件的播放,而且还可以播放MIDI 及CD 音频。
  *低级Wave 音频设备。用这些设备可以运行完全控制Wave 数据的应用文件。
  前两种方法使用简单,但无法实现两个及两个以上的WAV 文件的播放;第三种方法实现起来非常复杂,非专业人员很难完成。DirectSound 可以实现八个及八个以上WAV 文件的同时播放,能使多媒体程序更加生动,而且简单易用,但它仅适合于Windows 95 下的程序,也不支持WAV 文件的记录及存储-- 这是唯一让人遗憾的地方。
  实现DirectSound 需要以下几个步骤:
  1. 创建及初始化DirectSound:
LPDIRECTSOUND m_pDirectObject;
if(DirectSoundCreate(NULL,
  &m_pDirectObject,NULL)==DS_OK)
// 设定应用程序的声音 设备优先
  级别方式,一般为DSSCL_NORMAL
  m_pDirectObject ->SetCooperativeLevel(this
   ->m_hWnd,DSSCL_NORMAL);
else
   AfxMessageBox("DirectSound Create failed");
  2. 将WAV 文件读入内存,找到格式块、数据块位置及数据长度:
   m_pMemory,m_pFormat,m_pData,
   m_dwSize 在头文件中声明。
BOOL CDirectWave::LoadFile (CString Filename){
CFile File;
DWORD dwSize;
if (!File.Open (Filename, CFile::modeRead
  | CFile::shareDenyNone))
return FALSE;
dwSize = File.Seek (0, CFile::end);
File.Seek (0, CFile::begin);
//m_pMemory 内存存储块指针, 类型:LPVOID
m_pMemory = GlobalAlloc (GMEM_FIXED, dwSize);
if (File.ReadHuge (m_pMemory, dwSize) != dwSize){
   File.Close ();
   return FALSE;
   }
   File.Close ();
LPDWORD pdw,pdwEnd;
DWORD dwRiff,dwType, dwLength, dwLength;
if (m_pFormat) // 格式块指针,
        类型:LPWAVEFORMATEX
   m_pFormat = NULL;
if (m_pData) // 数据块指针,类型:LPBYTE
   m_pData = NULL;
if (m_dwSize) // 数据长度,类型:DWORD
   m_dwSize = 0;
pdw = (DWORD *) m_pMemory;
dwRiff = *pdw ++;
dwLength = *pdw ++;
dwType = *pdw ++;
if (dwRiff != mmioFOURCC ('R', 'I', 'F', 'F'))
   return FALSE;
if (dwType != mmioFOURCC ('W', 'A', 'V', 'E'))
   return FALSE;
// 寻找格式块,数据块位置及数据长度
pdwEnd = (DWORD *)((BYTE *)
     pdw +dwLength -4);
while (pdw < pdwEnd){
   dwType = *pdw ++;
   dwLength = *pdw ++;
   switch (dwType){
   case mmioFOURCC('f', 'm', 't', ' '):
   if (!m_pFormat){
   if (dwLength < sizeof (WAVEFORMAT))
     return FALSE;
     m_pFormat = (LPWAVEFORMATEX) pdw;
       if (m_pData &&m_dwSize)
     return TRUE;
     }
     break;
   case mmioFOURCC('d', 'a', 't', 'a'):
 if (!m_pData || !m_dwSize){
   m_pData = (LPBYTE) pdw;
   m_dwSize = dwLength;
   if (m_pFormat)
   return TRUE;
   }
   break;
 }
 pdw = (DWORD *)((BYTE *) pdw
     +((dwLength +1) &~1));
  }
  // 未找到,返回FALSE
  return FALSE;
}
  3. 创建声音缓冲区:
DSBUFFERDESC BufferDesc;
memset ( &BufferDesc, 0, sizeof (BufferDesc));
BufferDesc.lpwfxFormat =
    (LPWAVEFORMATEX) m_pFormat;
BufferDesc.dwSize = sizeof (DSBUFFERDESC);
BufferDesc.dwBufferBytes = m_dwSize;
BufferDesc.dwFlags = 0;
// 头文件中声明m_pDSoundBuffer,
  类型:LPDIRECTSOUNDBUFFER
if (pDSoundObject ->CreateSoundBuffer
 ( &BufferDesc, &m_pDSoundBuffer, 0) != DS_OK)
return FALSE;
  4. 载入声音数据:
BOOL CDirectWave::LoadData (void){
LPVOID lpPtr1, lpPtr2;
DWORD dwLen1, dwLen2;
HRESULT hResult;
TryLoad:
hResult = m_pDSoundBuffer ->Lock
(0, m_dwSize, &lpPtr1, &dwLen1,
  &lpPtr2, &dwLen2, 0);
if (hResult == DS_OK){
 memcpy (lpPtr1, m_pData, dwLen1);
 if (lpPtr2)
  memcpy (lpPtr2, m_pData +dwLen1, dwLen2);
  m_pDSoundBuffer ->Unlock
   (lpPtr1, dwLen1, lpPtr2, dwLen2);
  return TRUE;
  }
else if (hResult == DSERR_BUFFERLOST){
   hResult = m_pDSoundBuffer ->Restore ();
   if (hResult == DS_OK)
   goto TryLoad;
   }
return FALSE;
}
  5. 播放及停止:
void CDirectWave::Play (BOOL bLoop){
   DWORD dwFlags = 0;//dwFlags=
   DSBPLAY_LOOPING 可实现循环播放
TryPlay:
  if (m_pDSoundBuffer ->Play
  (0, 0, dwFlags) == DSERR_BUFFERLOST){
    if (LoadData ())// 数据丢失,重新装载
       goto TryPlay;
     }
}
void CDirectWave::Stop (void){
   m_pDSoundBuffer ->Stop ();
}
  6. 关闭、释放内存块、声音缓冲区。 

原文转自:http://www.ltesting.net