• 将声音文件加入VC


    概述 

      VC++是微软公司开发的一个IDE(集成开发环境),换句话说,就是使用c++的一个开发平台.有些软件就是这个编出来的...另外还有VB,VF.只是使用不同语言...但是,VC++是Windows平台上的C++编程环境,学习VC要了解很多Windows平台的特性并且还要掌握MFC、ATL、COM等的知识,难度比较大。Windows下编程需要了解Windows的消息机制以及回调(callback)函数的原理;MFC是Win32API的包装类,需要理解文档视图类的结构,窗口类的结构,消息流向等等;COM是代码共享的二进制标准,需要掌握其基本原理等等。 
      VC作为一个主流的开发平台一直深受编程爱好者的喜爱,但是很多人却对它的入门感到难于上青天,究其原因主要是大家对他错误的认识造成的,严格的来说VC++不是门语言,虽然它和C++之间有密切的关系,如果形象点比喻的话,可以把C++看作为一种“工业标准”,而VC++则是某种操作系统平台下的“厂商标准”,而“厂商标准”是在遵循“工业标准”的前提下扩展而来的。
      VC++应用程序的开发主要有两种模式,一种是WIN API方式,另一种则是MFC方式,传统的WIN API开发方式比较繁琐,而MFC则是对WIN API再次封装,所以MFC相对于WIN API开发更具备效率优势,但为了对WINDOWS开发有一个较为全面细致的认识,笔者在这里还是以讲解WIN API的相关内容为主线。
      话说到这里可能更多人关心的是学习VC++需要具备什么条件,为什么对于这扇门屡攻不破呢?
      要想学习好VC必须具备良好的C/C++的基础,必要的英语阅读能力也是必不可少的,因为大量的技术文档多以英文形式发布。

      本书主要从程序内部运行的机制和MFC程序的组织脉络入手,使读者在学习VC++编程知识时,既能够知其然,又能知其所以然,从而帮助读者从根本上理解和掌握Windows的程序设计。另外,全书还贯穿作者多年来学习编程的一些经验,以及一些学习方法的建议,为读者进一步的学习提供指导。

      本书从实际应用入手,由浅入深、循序渐进地讲述Windows程序内部运行机制、MFC框架、文本、菜单、对话框、文件操作、网络编程、进程间通信、ActiveX控件、动态链接库、HOOK编程等多个主题。

      本书内容丰富、实用性强,许多代码可以直接应用到工程项目中。书中的配套光盘还免费提供近乎45小时的VC++教学视频,读者在学习的过程中可以将视频和书互为参考,配合学习,这样可以更快、更好地掌握VC++编程。

      本书适合于VC++6.0的初学者和使用VC++从事开发的程序员,对于具有一定VC++编程经验的读者价值。

      VC++中播放声音的方法编辑本段  声音是多媒体的一个重要组成部分,在应用程序中加入声音可以使界面更友好。在VC++中可以根据不同的应用要求,用不同的方法实现声音的播放。 
      一.播放声音文件的简单方法 
      在VC++ 中的多媒体动态连接库中提供了一组与音频设备有关的函数。利用这些函数可以方便地播放声音。最简单的播放声音方法就是直接调用VC++中提供的声音播放函数BOOL sndPlaySound ( LPCSTR lpszSound,UINT fuSound ); 或BOOL PlaySound( LPCSTR lpszSound, HMODULE hmod, DWORD fuSound );其中参数lpszSound是需要播放声音的.WAV文件的路径和文件名, hmod在这里为NULL,fuSound是播放声音的标志,详细说明请参考VC++中的帮助。 例如播放C:soundmusic.wav可以用sndPlaySound ("c:soundmusic.wav",SND_ASYNC);或PlaySound("c:soundmusic.wav",NULL, SND_ASYNC|SND_NODEFAULT );如果没有找到music.wav文件,第一种格式将播放系统默认的声音,第二种格式不会播放系统默认的声音。 
      二.将声音文件加入到程序中 
      
    在VC++的程序设计中,可以利用各种标准的资源,如位图,菜单,对话框等。同时VC++也允许用户自定义资源,因此我们可以将声音文件作为用户自定义资源加入程序资源文件中,经过编译连接生成EXE文件,实现无.WAV文件的声音播放。 要实现作为资源的声音文件的播放,首先要在资源管理器中加入待播放的声音文件(实现过程并不复杂,这里不在叙述)。假设生成的声音文件资源标识符为IDR_WAVE1。在播放时只需要调用下面的语句: 
      PlaySound(MAKEINTRESOURCE(IDR_WAVE1),AfxGetResourceHandle(), SND_ASYNC|SND_RESOURCE|SND_NODEFAULT|SND_LOOP); 
      其中MAKEINTRESOURCE()宏将整数资源标识符转变为字符串,AfxGetResourceHandle()函数返回包含资源的模块句柄, 
      SND_RESOURCE是必须的标志。 
      作为资源的声音文件的第二种播放方法是把资源读入内存后作为内存数据播放。具体步骤入下: 
      1.获得包含资源的模块句柄: 
      HMODULE hmod=AfxGetResourceHandle(); 
      2.检索资源块信息: 
      HRSRC hSndResource=FindResource(hmod,MAKEINTRESOURCE(IDR_WAVE1),_T("WAVE")); 
      3. 装载资源数据并加锁: 
      HGLOBAL hGlobalMem=LoadResource(hmod,hSndResource); 
      LPCTSTR lpMemSound=(LPCSTR)LockResource(hGlobalMem); 
      4.播放声音文件: 
      sndPlaySound(lpMemSound,SND_MEMORY)); 
      5.释放资源句柄: 
      FreeResource(hGlobalMem); 
      三.播放声音文件的高级方法 
      
    在VC++中提供了一组对音频设备及多媒体文件直接进行操作的函数。利用这些函数可以灵活地对声音文件进行各种处理。 
      首先介绍几个要用到的数据结构。WAVEFORMATEX结构定义了WAVE音频数据文件的格式。WAVEHDR结构定义了波形音频缓冲区。读出的数据首先要填充此缓冲区才能送音频设备播放。WAVEOUTCAPS结构描述了音频设备的性能。MMCKINFO结构包含了RIFF文件中一个块的信息。详细的说明请参考VC++中的帮助。 
      下面给出程序流程简图及程序源代码清单,在VC++环境下可直接使用: 
      源程序清单如下: 

    LPSTR szFileName;//声音文件名 
    MMCKINFO mmckinfoParent; 
    MMCKINFO mmckinfoSubChunk; 
    DWORD dwFmtSize; 
    HMMIO m_hmmio;//音频文件句柄 
    DWORD m_WaveLong; 
    HPSTR lpData;//音频数据 
    HANDLE m_hData; 
    HANDLE m_hFormat; 
    WAVEFORMATEX * lpFormat; 
    DWORD m_dwDataOffset; 
    DWORD m_dwDataSize; 
    WAVEHDR pWaveOutHdr; 
    WAVEOUTCAPS pwoc; 
    HWAVEOUT hWaveOut; 
    //打开波形文件 
    if(!(m_hmmio=mmioOpen(szFileName,NULL,MMIO_READ|MMIO_ALLOCBUF))) 
    { 
        //File open Error 
        Error("Failed to open the file.");//错误处理函数 
        return false; 
    } 
    //检查打开文件是否是声音文件 
    mmckinfoParent.fccType =mmioFOURCC(’W’,’A’,’V’,’E’); 
    if(mmioDescend(m_hmmio,(LPMMCKINFO)&mmckinfoParent,NULL,MMIO_FINDRIFF)) 
    { 
        //NOT WAVE FILE AND QUIT 
    } 
    //寻找 ’fmt’ 块 
    mmckinfoSubChunk.ckid =mmioFOURCC(’f’,’m’,’t’,’ ’); 
    if(mmioDescend(m_hmmio,&mmckinfoSubChunk,&mmckinfoParent,MMIO_FINDCHUNK)) 
    { 
        //Can’t find ’fmt’ chunk 
    } 
    //获得 ’fmt ’块的大小,申请内存 
    dwFmtSize=mmckinfoSubChunk.cksize ; 
    m_hFormat=LocalAlloc(LMEM_MOVEABLE,LOWORD(dwFmtSize)); 
    if(!m_hFormat) 
    { 
        //failed alloc memory 
    } 
    lpFormat=(WAVEFORMATEX*)LocalLock(m_hFormat); 
    if(!lpFormat) 
    { 
        //failed to lock the memory 
    } 
    if((unsigned long)mmioRead(m_hmmio,(HPSTR)lpFormat,dwFmtSize)!=dwFmtSize) 
    { 
        //failed to read format chunk 
    } 
    //离开 fmt 块 
    mmioAscend(m_hmmio,&mmckinfoSubChunk,0); 
    //寻找 ’data’ 块 
    mmckinfoSubChunk.ckid=mmioFOURCC(’d’,’a’,’t’,’a’); 
    if(mmioDescend(m_hmmio,&mmckinfoSubChunk,&mmckinfoParent,MMIO_FINDCHUNK)) 
    { 
        //Can’t find ’data’ chunk 
    } 
    //获得 ’data’块的大小 
    m_dwDataSize=mmckinfoSubChunk.cksize ; 
    m_dwDataOffset =mmckinfoSubChunk.dwDataOffset ; 
    if(m_dwDataSize==0L) 
    { 
        //no data in the ’data’ chunk 
    } 
    //为音频数据分配内存 
    lpData=new char[m_dwDataSize]; 
    if(!lpData) 
    { 
        //faile 
    } 
    if(mmioSeek(m_hmmio,SoundOffset,SEEK_SET)<0) 
    { 
        //Failed to read the data chunk 
    } 
    m_WaveLong=mmioRead(m_hmmio,lpData,SoundLong); 
    if(m_WaveLong<0) 
    { 
        //Failed to read the data chunk 
    } 
    //检查音频设备,返回音频输出设备的性能 
    if(waveOutGetDeVCaps(WAVE_MAPPER,&pwoc,sizeof(WAVEOUTCAPS))!=0) 
    { 
        //Unable to allocate or lock memory 
    } 
    //检查音频输出设备是否能播放指定的音频文件 
    if(waveOutOpen(&hWaveOut,DevsNum,lpFormat,NULL,NULL,CALLBACK_NULL)!=0) 
    { 
        //Failed to OPEN the wave out devices 
    } 
    //准备待播放的数据 
    pWaveOutHdr.lpData =(HPSTR)lpData; 
    pWaveOutHdr.dwBufferLength =m_WaveLong; 
    pWaveOutHdr.dwFlags =0; 
    if(waveOutPrepareHeader(hWaveOut,&pWaveOutHdr,sizeof(WAVEHDR))!=0) 
    { 
        //Failed to prepare the wave data buffer 
    } 
    //播放音频数据文件 
    if(waveOutWrite(hWaveOut,&pWaveOutHdr,sizeof(WAVEHDR))!=0) 
    { 
        //Failed to write the wave data buffer 
    } 
    //关闭音频输出设备,释放内存 
    waveOutReset(hWaveOut); 
    waveOutClose(hWaveOut); 
    LocalUnlock(m_hFormat); 
    LocalFree(m_hFormat); 
    delete [] lpData; 


      说明:1)以上使用的音频设备和声音文件操作函数的声明包含在mmsystem.h头文件中,因此在程序中必须用#include "mmsystem.h"语句加入头文件。同时在编译时要加入动态连接导入库winmm.lib,具体实现方法是从Developer Studio的Project菜单中选择Settings,然后在Link选项卡上的Object/Library Modules控制中加入winmm.lib。2)在pWaveOutHdr.lpData中指定不同的数据,可以播放音频数据文件中任意指定位置的声音。3) 以上程序均在VC++6.0中调试通过,在文中省略了对错误及异常情况的处理,在实际应用中必须加入。 
      四.结论 
      在VC++中可以根据应用需要采用不同的方法播放声音文件。简单应用可以直接调用声音播放函数。第二种方法可以把声音作为资源加入可执行文件中。如果在播放之前要对声音数据进行处理,可用第三种方法。 
      五.Visual C++.NET
      随着NET的时代来临,VC++也如同VB一样,脱胎换骨到VC++.NET

  • 相关阅读:
    如何解决秒杀的性能问题和超卖的讨论
    Redis作者谈Redis应用场景
    Net开源工作流Roadflow的使用与集成
    VisualStudioOnline协同工作流程
    中文乱码?不,是 HTML 实体编码!(转)
    Nginx搭建反向代理服务器过程详解(转)
    2016年1月15日面试某互联网公司总结(转)
    浏览器缓存知识小结及应用(转)
    Gradle多项目配置的一个demo
    Android getResources的作用和须要注意点
  • 原文地址:https://www.cnblogs.com/MrYuan/p/6001012.html
Copyright © 2020-2023  润新知