只是使用的假设PlaySound()这个API函数来显示的声音效果,然后,然后,它不会出现在混合声音,因为PlaySound还有播放期间声音,这将不可避免地导致现有声音停止。

因此,使用

PlaySound()包括丰富音乐与音效的游戏世界是不现实的。
    而DirectSound就能够完美的解决混音问题,并且它直接针对硬件编程。最大程度上减小了游戏进程逻辑对于声音播放效果的影响。
    在这篇文章里,先谈谈DirectSound的使用。

初始化工作第一步,也是全部DirectX组建初始化的必做工作,总的来说分为三个步骤:1、设置好库连接的支持。2、加入须要的头文件。

3、也是大家最easy遗忘的一步,将

设置文件夹中与DirectX SDK相关的库与头文件的连接保持在最顶端。
    以下针对DirectSound来具体讲讲上面三个步骤,1.加入库连接,能够有两种方法,你能够在project菜单条中选择设置选项,并在连接这一项的对象与库模块这一栏目中写上

Dsound.lib Dxguid.lib两个字符串。你也能够仅仅在工作区用加入文件的方法把这两项放在里面。2.加入<dsound.h>头文件放在代码段的顶端。这个无须赘言。3.假设此时发现有些

Directsound的类名无法识别,那么请检查工具菜单项的设置选项中文件夹的设置,看是否在lib与include中都将DirectX SDK相关内容都放在第一位,由于在编译连接中第一位的库

是作为连接的首选的。
    另外还应加入Winmm.lib以及mmsystem.h,mmreg.h头文件,由于载入WAVE文件时会用到。

初始化工作第二步,DirectSound对象的建立
    (1)建立DirectSound对象
    (2)设定共享层级
    (3)设定主缓冲区的格式
    首先要建立一个代表声卡的DirectSound对象,我们先定义LPDIRECTSOUND pDS,然后用DirectSoundCreate(NULL, &pDS, NULL);方法来建立它,假设你想察看这个对象是否成功

的建立,能够定义一个HRESULT result变量。由它接受DirectSoundCreate方法的返回值得并推断是否等于DS_OK,假设不等于它,则能够用MessageBox()方式弹出对话框来告诉程

序员建立失败。注意DirectSoundCreate()中的第一个參数。是NULL。表示使用眼下预设的声卡。也能够调用DirectSoundEnumerate取得可用的声卡。
    然后要设定程序协调层级,使用pDS调用SetCooperativeLevel方法来实现,注意这种方法有两个參数,第一个參数代表应用程序的主窗体,而第二个參数则设定使用资源的优

先权。
    最后要看看缓冲区的概念。主缓冲区能够看作一个DirectSound是用来播放声音,产生混音效果的区域。它能够自己主动生成。也能够自己建立,但假设自己建立并设定其播放模式

,在设置协调层级时,标志位必须设定为DSSCL_PRIORITY.次缓冲区则存储播放声音的文件。在载入声音文件后,仅仅要调用Play()方法,声音就会自己主动的送入主缓冲区中并进行播放

在初始化过程中。应重点注意DSBUFFERDESC结构,它担负着区分主次缓冲区以及缓冲区明细初始化的重任,在使用它时,首先要清空,能够使用memset()方法来将其全部内存中

的位设为0。同一时候要设置结构的大小,并确定它的标志位,以及设置缓冲区大小与格式,其详细的初始化过程能够看文章结尾的样例。

在完毕了初始化工作后。应该先把须要播放的声音文件加载到已经完毕初始化的次缓冲区中。这里重点讲下怎样读入一个声音文件以及取得当中的信息与播放的资料。

首先我们要知道,WAVE是利用区块(chunk)方式存储文件的,包括纪录文件格式的fmt区块与文件实际内容的data区块。因此读取文件必需要完毕下面步骤:
    (1)打开文件
    (2)确认是否为RIFF文件。类型为WAVE
    (3)寻找fmt区块,取得文件格式
    (4)寻找data区块。取得文件内容
    (5)关闭文件
    (6)载入声音入次缓冲区
详细过程见样例。

最后当然是播放与停止的使用了,详细能够自己去用次缓冲区指针试一下。
/*--------------------------------------------------------------------------------*/

//    以下是我写的一个使用DirectSound的样例:
下面为头文件部分
#ifndef GAMESOUND
#define GAMESOUND

#include "dsound.h"
#include "windows.h"
#include "mmsystem.h"
#include "mmreg.h"

class GameSound
{
private:
    HWND soundhwnd;

HRESULT result;      //用来接受建立后的返回值
    LPDIRECTSOUND pDS;   //代表声卡的DirectSound对象
    LPDIRECTSOUNDBUFFER pMainBuf;  //声明主缓冲区指针
    DSBUFFERDESC desc;             //声明描写叙述结构。用来初始化缓冲区域
    WAVEFORMATEX pwfmt;            //声明声音结构,用来设定播放格式
 
    WAVEFORMATEX swfmt;    //声明声音结构
    MMCKINFO     ckRiff;   //RIFF区块的信息
    MMCKINFO     ckInfo;   //子区块的信息
    MMRESULT     mmresult; //返回的结果
    DWORD        size;     //实际资料的大小
    HMMIO        hbackground;    //打开的多媒体文件
public:
    GameSound();
    void GameSoundInit(HWND);             //GameSound对象的建立
    void GameSoundbufferConstruct();      //缓冲区的创建
    void GameSoundfmtSet(int ,int ,int);  //通过主缓冲区指针来设置播放格式
    void GameSoundReadWAVfile(char*, HMMIO&);//将声音文件读入并将明细存在HMMIO结构中
    void GameSoundReadinbuffer(LPDIRECTSOUNDBUFFER&, char*);//将声音文件读入次缓冲区中

LPDIRECTSOUNDBUFFER pStartmusic;  //声明子缓冲区指针(開始音乐指针)
    LPDIRECTSOUNDBUFFER pTalkmusic;  //声明子缓冲区指针(谈天音乐指针)
    LPDIRECTSOUNDBUFFER pWalkmusic;  //声明子缓冲区指针(行走音乐指针)
    LPDIRECTSOUNDBUFFER pWarmusic;  //声明子缓冲区指针(战斗音乐指针)
    LPDIRECTSOUNDBUFFER pyudimusic;  //声明子缓冲区指针(攻击声音指针)
    LPDIRECTSOUNDBUFFER pwinmusic;  //声明子缓冲区指针(胜利音乐指针)
    LPDIRECTSOUNDBUFFER plosemusic;  //声明子缓冲区指针(失败声音指针)

LPDIRECTSOUNDBUFFER pAttacksound;  //声明子缓冲区指针(攻击声音指针)
    LPDIRECTSOUNDBUFFER pAIAttacksound;  //声明子缓冲区指针(攻击声音指针)

void GameSoundAllstop();  //for背景音乐。让背景音乐更换时,先前的全部音乐都停止,从而播放新的音乐
    void GameMusicplay(LPDIRECTSOUNDBUFFER&);     //用来播放循环音乐
    void GameSoundplay(LPDIRECTSOUNDBUFFER&);     //用来播放一次性音效
};

#endif

下面为源文件部分
#include "GameSound.h"

GameSound::GameSound()
{
}

void GameSound::GameSoundInit(HWND hwnd)
{
 this->pDS;
 this->soundhwnd = hwnd;
 this->result = DirectSoundCreate(NULL, &pDS, NULL);
 if(this->result != DS_OK)
  MessageBox(hwnd, "建立 DirectSound 对象失败!", NULL,MB_OK);

this->result = this->pDS->SetCooperativeLevel(hwnd, DSSCL_PRIORITY);
 if(this->result != DS_OK)
  MessageBox(hwnd, "设定程序协调层级失败!", NULL,MB_OK);

this->GameSoundbufferConstruct(); 
}

void GameSound::GameSoundbufferConstruct()
{

memset(&this->desc, 0, sizeof(desc));   //清空结构内容
 desc.dwSize = sizeof(desc);             //配制描写叙述结构大小
 desc.dwFlags = DSBCAPS_PRIMARYBUFFER;   //???
 desc.dwBufferBytes = 0;
 desc.lpwfxFormat = NULL;
 result = pDS->CreateSoundBuffer(&desc, &this->pMainBuf, NULL);
 if(this->result != DS_OK)
  MessageBox(this->soundhwnd, "建立主缓冲区域失败!", NULL,MB_OK);

this->GameSoundReadinbuffer(this->pTalkmusic, "sound//talk2.wav");
 this->GameSoundReadinbuffer(this->pStartmusic, "sound//startwav.wav");
 this->GameSoundReadinbuffer(this->pWarmusic, "sound//zhandou.wav");
 this->GameSoundReadinbuffer(this->pWalkmusic, "sound//mainwav.wav");
 this->GameSoundReadinbuffer(this->pAttacksound, "sound//fire.wav");
 this->GameSoundReadinbuffer(this->pAIAttacksound, "sound//fire2.wav");
 this->GameSoundReadinbuffer(this->pyudimusic, "sound//yudi.wav");
 this->GameSoundReadinbuffer(this->pwinmusic, "sound//win.wav");
 this->GameSoundReadinbuffer(this->plosemusic, "sound//lose.wav");
 
}

void GameSound::GameSoundfmtSet(int channels, int SamplesPerSec, int wBitPerSample)
{
    memset(&this->pwfmt, 0, sizeof(pwfmt));
    this->pwfmt.wFormatTag = WAVE_FORMAT_PCM;
    this->pwfmt.nChannels = channels;
    this->pwfmt.nSamplesPerSec = SamplesPerSec;
    this->pwfmt.wBitsPerSample = wBitPerSample;
    this->pwfmt.nBlockAlign = this->pwfmt.wBitsPerSample / 8 * this->pwfmt.nChannels;
    this->pwfmt.nAvgBytesPerSec = this->pwfmt.nSamplesPerSec * this->pwfmt.nBlockAlign;
    this->result = this->pMainBuf->SetFormat(&this->pwfmt);
    if(this->result != DS_OK)
 MessageBox(this->soundhwnd, "设定播放格式失败!

", NULL,MB_OK); 
}

void GameSound::GameSoundReadWAVfile(char* filename, HMMIO &hmmbackground)
{
 hmmbackground = mmioOpen(filename, NULL, MMIO_ALLOCBUF | MMIO_READ);  //打开文件
 if(hmmbackground == NULL)
  MessageBox(this->soundhwnd, "文件不存在!", NULL,MB_OK);

//搜索类型
 ckRiff.fccType = mmioFOURCC('W', 'A', 'V', 'E');//设定文件类型
 mmresult = mmioDescend(hmmbackground, &ckRiff, NULL, MMIO_FINDRIFF);
 if(mmresult != MMSYSERR_NOERROR)
        MessageBox(this->soundhwnd, "文件格式错误!

", NULL,MB_OK);

//搜索区块
 ckInfo.ckid = mmioFOURCC('f', 'm', 't', ' ');//设定区块类型
 mmresult = mmioDescend(hmmbackground, &ckInfo, &ckRiff, MMIO_FINDCHUNK);
        if(mmresult != MMSYSERR_NOERROR)
  MessageBox(this->soundhwnd, "文件格式错误!", NULL,MB_OK);
 if(mmioRead(hmmbackground, (HPSTR)&swfmt, sizeof(swfmt)) == -1)
  MessageBox(this->soundhwnd, "读取格式失败!

", NULL,MB_OK);

mmresult = mmioAscend(hmmbackground, &ckInfo, 0);   //跳出子区块

//搜索区块
 ckInfo.ckid = mmioFOURCC('d', 'a', 't', 'a');
 mmresult = mmioDescend(hmmbackground, &ckInfo, &ckRiff, MMIO_FINDCHUNK);
 if(mmresult != MMSYSERR_NOERROR)
  MessageBox(this->soundhwnd, "文件格式错误!", NULL,MB_OK);

size = ckInfo.cksize;

}

void GameSound::GameSoundReadinbuffer(LPDIRECTSOUNDBUFFER& buffer, char* filename)
{
 LPVOID pAudio;
 DWORD bytesAudio;

this->GameSoundReadWAVfile(filename, this->hbackground);

memset(&this->desc, 0, sizeof(desc));   //清空结构内容
 desc.dwSize = sizeof(desc);             //配制描写叙述结构大小
 desc.dwFlags = DSBCAPS_STATIC | DSBCAPS_CTRLPAN |
             DSBCAPS_CTRLVOLUME | DSBCAPS_GLOBALFOCUS;   //?

??
 desc.dwBufferBytes = this->size;
 desc.lpwfxFormat = &this->swfmt;
 result = pDS->CreateSoundBuffer(&desc, &buffer, NULL);
 if(this->result != DS_OK)
  MessageBox(this->soundhwnd, "建立次缓冲区域失败!

", NULL,MB_OK);

result = buffer->Lock(0, this->size, &pAudio, &bytesAudio, NULL, NULL, NULL);
 if(this->result != DS_OK)
  MessageBox(this->soundhwnd, "锁定缓冲区失败!", NULL,MB_OK);

this->mmresult = mmioRead(this->hbackground, (HPSTR)pAudio, bytesAudio);

if(mmresult == -1)
  MessageBox(this->soundhwnd, "读取声音文件资料失败", NULL,MB_OK);

this->result = buffer->Unlock(pAudio, bytesAudio, NULL, NULL);

if(this->result != DS_OK)
  MessageBox(this->soundhwnd, "解除锁定缓冲区失败!", NULL,MB_OK);

mmioClose(this->hbackground, 0);
}

void GameSound::GameSoundAllstop()
{
 this->pAttacksound->Stop();
 this->pStartmusic->Stop();
 this->pTalkmusic->Stop();
 this->pWalkmusic->Stop();
 this->pWarmusic->Stop();
 this->pyudimusic->Stop();
 this->pwinmusic->Stop();
 this->plosemusic->Stop();
 this->pAIAttacksound->Stop();
 this->pAttacksound->SetCurrentPosition(0);
 this->pStartmusic->SetCurrentPosition(0);
 this->pTalkmusic->SetCurrentPosition(0);
 this->pWalkmusic->SetCurrentPosition(0);
 this->pWarmusic->SetCurrentPosition(0);
 this->pyudimusic->SetCurrentPosition(0);
 this->pwinmusic->SetCurrentPosition(0);
 this->plosemusic->SetCurrentPosition(0);
 this->pAIAttacksound->SetCurrentPosition(0);
}

void GameSound::GameMusicplay(LPDIRECTSOUNDBUFFER& buffer)
{
 this->GameSoundAllstop();

buffer->Play(0, 0, 1);
}

void GameSound::GameSoundplay(LPDIRECTSOUNDBUFFER& buffer)
{
 buffer->Play(0, 0, 0);
}

DirectSound应用相关推荐

  1. DirectSound的应用

    假设仅仅使用PlaySound()这个API函数来表现声音效果的话,那么就无法表现出声音的混音效果,由于PlaySound在播放还有一个声音时,必定会导致现有声音的停止.因此,使用 PlaySound ...

  2. 利用Directsound编程实现实时混音

    在游戏开发中比较常用的音效素材都是比较短的,所以一般常用的API是playsound()函数,比如我们要在游戏背景中播放一个test.wav音效素材,只要简单的调用下面的函数即可 PlaySound( ...

  3. 在 Delphi 下使用 DirectSound (12): 测试失真效果器 IDirectSoundFXDistortion8

    为什么80%的码农都做不了架构师?>>>    {相关结构:} TDSFXDistortion = packed recordfGain: Sinle; //-60 .. 0 : - ...

  4. 在 Delphi 下使用 DirectSound (5): 获取或设置缓冲区的格式:

    次缓冲区(或叫辅助缓冲区)尽管使用了波形文件自己的 TWaveFormatEx, 但最终播放的却只是 22050HZ 的 8 位立体声. 因为次缓冲区最终要混入主缓冲区才播放, 可主缓冲区的缺省格式是 ...

  5. 在 Delphi 下使用 DirectSound (9): 效果器初步及 IDirectSoundFXGargle8 效果器

    只有使用 IDirectSoundBuffer8 的次缓冲区才能设置"特效", 主缓冲区主要负责的是混音和处理 3D 效果. IDirectSoundBuffer8(非 IDire ...

  6. 在 Delphi 下使用 DirectSound (4): 设置音量、相位、播放频率和播放位置

    通过 IDirectSoundBuffer 的 SetVolume.SetPan.SetFrequency.SetCurrentPosition 方法可以简单进行这些设置. 同时 IDirectSou ...

  7. DirectSound学习笔记(3):协作级别

     协作级别 因为Windows是一个多任务环境,多个应用程序可能在任意时刻对一个设备驱动器进行操作.虽然使用协作级别,DirectX仍然确保每个应用程序不能以错误方式或在错误时刻访问设备.每个Dire ...

  8. DirectSound 混音的实现

    啥叫混音呢,其实很简单,如果两个人同时说话 ,他们俩发出的声波在空气中进行了波的叠加,这其实就是个混音.计算机的混音,其实是一个虚拟的混音操作,因为计算机其实是只有一个声源(现在的计算机通常有两声道甚 ...

  9. 在 Delphi 下使用 DirectSound (14): 测试镶边效果器 IDirectSoundFXFlanger8

    {相关结构:} TDSFXFlanger = packed recordfWetDryMix: Single; // 0 .. 100 : 50 (%)fDepth: Single; // 0 .. ...

  10. DirectSound开发

    获取语音设备并创建Interface       DirectSound开发中常常需要获取系统当前可用的语音设备列表,在SDK Samples中最先介绍了它.DirectX将它封装成函数DirectS ...

最新文章

  1. python使用TSNE为影像组学(radiomics)数据进行降维可视化分析
  2. BZOJ1150[CTSC2007]数据备份Backup——模拟费用流+堆+链表
  3. 未能加载文件或程序集“*****.dll”或它的某一个依赖项。找到的程序集清单定义与程序集引用不匹配。(异常来自HRESULT:0x80131040)
  4. AdapterView及其子类之一:基本原理(ListView、ListActivity类型)
  5. springboot:BeanPostProcessor示例及分析
  6. 用ShopEx网上开店之安装Zend插件[2]
  7. qt linux下实现vlc视频,vlc音视频开发(一)环境搭建(qt篇)
  8. 多继承、经典类与新式类、新式类的C3算法详解
  9. 创建Maven web工程不能解析EL表达式的解决办法
  10. oracle数据库一些常用数据库类型
  11. tp6 多关联withJoin查询
  12. 电商网站后台九大功能模块详解
  13. 堆排序(排升序为啥建大堆,排降序为啥建小堆)
  14. Java小游戏,防止物体跑出四周边界的算法
  15. N-Tiers开发方式(COM+组件的注册、修改)
  16. 车机蓝牙通话流程分析的流程分析
  17. 资源者、配置者、投资人
  18. Python简单实现ATM自动存取款机
  19. 51单片机最小组成系统电路图[1080p高清]
  20. 用DW制作简单的浮动广告

热门文章

  1. 开发小程序需要服务器吗?小程序服务器配置要求
  2. python实训名片管理程序_python实现名片管理系统
  3. 心蓝12306订票助手
  4. Python图像识别-Opencv02 二值图像、灰度图像以及彩色图像
  5. Office 2.0是什么?
  6. anz的swift code_澳洲Commonwealth bank是不是只有一个SWIFT CODE?
  7. Windows XP SP3 IIS HTTP 500 - 内部服务器错误解决
  8. oracle数据库merge into,merge into 的用法
  9. Multi-Label Image Classification(多标签图像分类)
  10. 排列 组合 算法(一)