几个API不会用(300分,不够再加)(两位朋友的回答都没用,问题未解决) ( 积分: 300 )

  • 主题发起人 主题发起人 newditter
  • 开始时间 开始时间
N

newditter

Unregistered / Unconfirmed
GUEST, unregistred user!
waveOutGetPitch &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;如何查询一个波形输出设备的当前音调设置 (100分)<br><br>如何获取游戏杆的所有的ID中的“游戏杆设备”名称, (200分)<br><br>请问:如何获取“波形输入”的音量,我在线等<br><br>下面的问题另计分(100分):<br>请问,WaveOutGetDevCaps 有API可以获取并设置音量,但“WaveINGetDevCaps”却没有相关的API获取当前的音量,<br>难度要获取“WaveINGetDevCaps”当前的音量,也是要用“WaveOutGetVolume”!<br><br>请高手解释一下或写出源代码(获取“WaveINGetDevCaps”当前的音量)
 
waveOutGetPitch &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;如何查询一个波形输出设备的当前音调设置 (100分)<br><br>如何获取游戏杆的所有的ID中的“游戏杆设备”名称, (200分)<br><br>请问:如何获取“波形输入”的音量,我在线等<br><br>下面的问题另计分(100分):<br>请问,WaveOutGetDevCaps 有API可以获取并设置音量,但“WaveINGetDevCaps”却没有相关的API获取当前的音量,<br>难度要获取“WaveINGetDevCaps”当前的音量,也是要用“WaveOutGetVolume”!<br><br>请高手解释一下或写出源代码(获取“WaveINGetDevCaps”当前的音量)
 
声卡编程讲解 &nbsp;<br>此文是转贴,如违反了你的版权,请email给我,我会立刻删除。<br><br>一、特点<br><br>声卡作为语音信号与计算机的接口卡件,其最基本的一项功能就是A/D转换。实际,除了语音外,很多信号的频率都落在音频范围内(比如机械量信号,过程量信号等,当我们需要对这些信号进行采集时,使用声卡作为采集卡是一种相当令人满意的解决方 案,其理由如下:<br><br>1. 价格便宜。一般声卡的价格才一百多元,比起自己从头到尾开发一块采集卡的成本 低 得多。比起目前市场上的采集卡的价格,更是不可同日而语。相应地,产品成本也会降低。<br><br>2. 即买即用。完全省略了A/D卡的的硬件开发过程,很大程度上缩短了产品的开发周期<br><br>3. 灵活性好。量化位数可编程(8位或16位);采样频率可编程(一般声卡的最高采样频率可达200KHz,并且连续可调);采样通道可编程(1通道或2通道);由于可以使用在Windows操作系统下,可以用通用的软件开发工具对其进行开发(如Delphi,VB,VC等)。<br><br>当然也有其局限性,那就是声卡一般只能作为PC插卡用在PC机上,很难用微控制器对其进行控制(因为要用到中断和DMA技术),因而很难用在小型的仪器仪表里。再者由于声卡是专门针对音频信号而设计的,所以它的采样速率不可能很高。<br><br>二、编程技术<br><br>既然声卡本身就是一块很好的A/D卡,硬件部分已经不需操心了,那么最终的问题就是怎样对其编程才能够取得A/D的数据。<br><br>事实上,声卡是PC的一种多媒体设备,所以可以用Windows 的MCI(Media Control I nterface)命令来控制声卡。MCI它提供了一组与设备无关的控制命令,是一种访问多媒体设备的高层次方法。也正因为它属于一种高层次方法,所以它提供给程序员的灵活性有限,利用MCI命令来控制声卡录音时,程序员不能在录音的过程中访问内存中的采样数据,只有在录音完成后通过访问*.WAV文件才可以得到采样数据,尽管最终还是得到了采样数据,但是这样做一方面嫌其麻烦,更重要的是存取文件需要耗费时间,声卡在采样的过程中有可能会停止下来等待文件操作,造成了采样的断续。在一些实时性要求比较高的场合(比如波形分析,实时控制等),断续的采样明显是不行的。 Windows的低级波形音频函数提供了对声卡的最大灵活性的操作,它允许在采样过程中随机地访问内存中的每个采样数据,完全可以克服使用MCI命令所遇到的实时性问题。 Windows以动态连接库Mmsystem.dll的形式提供低级波形音频函数,在Mmsystem.dll中总共包括了以下几个有关波形录入的函数:<br><br>waveInAddBuffer :向声音输入设备发送缓冲区 ;waveInClose :关闭声音输入设备<br>waveInGetDevCaps:获取声音输入设备性能; waveInGetErrorText:获取声音出错信息文本<br>waveInGetID :获取声音输入设备ID; waveInGetNumDevs:返回声音输入设备数量<br>waveInGetPosition :获取声音设备输入位置; waveInMessage :向声音输入设备发送信息<br>waveInOpen :打开声音输入设备; waveInPrepareHeader:预备声音输入缓冲区<br>waveInReset :停止声音输入设备工作; waveInStart :停止声音输入设备工作<br>waveInStop :停止声音输入; waveInUnprepareHeader : 清除预备的声音文件头<br><br>需要说明的是:不同的编程工具多会含有对这些低级波形音频函数进行说明的头文件(比如在Delphi4.0中,对Mmsystem.dll说明的文件是Mmsystem.pas),所以在不同的编程工具中调用这些函数时有可能会使用不同的名称。<br>与使用其他设备一样,要想用波形音频函数来控制声卡,必须要经过以下的步骤:<br>1. 打开波形输入设备。函数waveInOpen用于打开波形输入设备,其原型如下:<br>① WORD waveInOpen(lphWaveIn,wDeviceID,lpFormat,dwCallback,dwCallbackInstan ce, dwFlags) LPHWaveIn: lphWaveIn 该变量用来接收波形输入设备的句柄,该句柄应当保存下来,因为其他的波形输入函数还会用到它.<br>②WORD wDeviceID 该变量用来指明波形输入设备的标记号.当PC中有多块声卡(准确地说是波形输入设备)时,操作系统会为每一块声卡分配一个标记号.可以用waveInGetNumDev s函数来得到能够作为波形输入设备的数目N,则wDeviceID的取值范围为0~N-1.如果想得到没个标记号所对应的录音性能,可以使用函数waveInGetDevCaps.若把wDeviceID设为W AVE_MAPPER(即-1),则系统会自动选择一符合要求的设备(根据lpFormat的要求). ③lpFormat是一个指向PCMWAVEFORMAT数据结构的指针,应当在这个数据结构中指明所期望的采样模式,这个数据结构的定义是这样的:<br>Typedef structure pcmwaveformat_tag { WAVEFORMAT wf; //有关PCM格式设置的另外一种数据结构<br>WORD wBitsPerSample; //量化位数<br>}PCMWAVEFORMAT;<br>Typedef structure waveformat_tag {<br>WORD wFormatTag; //采样数据格式,目前只能用PCM格式<br>WORD nChannels; //通道数目(1或2)<br>DWORD nSamplesPerSec; //采样速率<br>DWORD nAvgBytesPerSec;//每秒采样得到的数据<br>WORD nBlockAlign; //记录区块对齐的单位。此值为nChannels*wBitsPerSample/8 }WAVEFORMAT;<br>④ DWORD dwCallback.定义回调函数的地址或回调窗口的句柄。回调函数的地址或回调窗口用来处理波形输入设备产生的消息。<br>⑤DWORD dwCallbackInstance。这是一个用户自定义的数据,该数据会一并传给回调函数(或窗口)。<br>⑥DWORD dwFlags。定义打开波形输入设备的标记。<br>CALLBACK_WINDOW 定义dwCallback为窗口句柄。<br>CALLBACK_FUNCTION 定义dwCallback为函数地址。<br>另外还可以在此指定:<br>WAVE_FORMAT_QUERY 只查询波形输入设备是否支持给定格式而不真的打开波形输入设备。<br>WAVE_ALLOWSYNC 同步方式开启波形输入设备,录音工作在后台进行。<br>下面一段Delphi程序说明了打开波形输入设备的过程:<br><br>type<br>TRecorder = class<br>private<br>…<br>FWaveFmt : TWaveFormatEx;//Delphi中,WAVEFORMAT和PCMWAVEFORMAT合为TwaveFor<br>matEx。<br>WaveHandle : HWaveIn;<br>WaveHdr1 : PWAVEHDR; //数据缓冲区头结构的指针 (见下文)<br>WaveBuffer1 : lpstr; //数据缓冲区的指针 (见下文)<br>procedure CallBack(uMsg,dwInstance,dwParam1,dwParam2 : DWORD); stdcall;<br>…<br>end;<br>…<br>Recorder:=TRecorder.Create;<br>…<br>Recorder.FWaveFmt.wFormatTag:=WAVE_FORMAT_PCM;<br>Recorder.FWaveFmt.wBitsPerSample:=16;<br>Recorder.FWaveFmt.nSamplesPerSec:=11025;<br>Recorder.FWaveFmt.nAvgBytesPerSec:=22050;<br>Recorder.FWaveFmt.nBlockAlign:=2;<br>WaveInOpen(@Recorder.WaveHandle,Wave_Mapper,mailto:@Recorder.FWaveFmt,<br>DWORD(@TRecorder.CallBack),DWORD(@Recorder),CALLBACK_FUNCTION + WAVE_ALLOW<br>SYNC);<br>…<br><br>2. 为采样数据分配缓冲空间<br><br>在Windows环境,可以用GlobalAllocPtr来获取一段内存空间,但是由于Windows操作系统采用了虚拟存储管理机制,这块内存空间随时有可能会被置换到硬盘上,读写硬盘所耗费的时间会造成采样的不连续。因此,在将缓冲区送往波形输入设备之前,必须调用WaveInPrepareHeader函数以保证缓冲区不会被置换到硬盘上。当然在用GlobalFreeP tr来释放缓冲区之前,必须先要用WaveInUnprepareHeader函数来解除这种保护。<br><br>下面几行Delphi语句说明了使用录音缓冲区的过程。<br><br>…<br>Recorder.WaveHdr1:=GlobalAllocPtr(GHND or GMEM_SHARE,Sizeof(WAVEHDR));<br>Recorder.WaveBuffer1:=GlobalAllocPtr(GHND or GMEM_SHARE,1024);<br>Recorder.WaveHdr1.lpData := Recorder.WaveBuffer1;<br>Recorder.WaveHdr1.dwBufferLength:=1024;<br>WaveInPrepareHeader(Recorder.WaveHandle, Recorder.WaveHdr1, sizeof(WAVEHDR<br>));<br>WaveInAddBuffer(Recorder.WaveHandle, Recorder.WaveHdr1, sizeof(WAVEHDR));<br>…<br>WaveInUnprepareHeader(Recorder.WaveHandle, Recorder.WaveHdr1, sizeof(TWAVE<br>HDR));<br>GlobalFreePtr(Recorder.WaveBuffer1);<br>…<br><br>但是,如果只为波形输入设备开辟一个缓冲区,则当该缓冲区被采样数据填满后,波形输入设备就无缓冲区可用,不得不停止采样,从而造成了采样的断续。所以在实际应用中,至少要为波形输入设备准备两个缓冲区,用上述方法同时送给波形输入设备。<br><br>3. 启动波形输入设备<br><br>当上述一切都准备好后,用WaveInStart启动波形输入设备,即可开始进行数据采集,在采集的过程中,一旦有缓冲区被采样数据填满,系统就回调WaveInOpen中指定的dwCa llback函数(或向指定的窗口发送消息)。在Delphi4.0中,回调函数的格式是这样的: procedure CallBack(uMsg,dwInstance,dwParam1,dwParam2 : DWORD); stdcall; 其中uMsg是Windows的消息标记号,有三种情况:<br><br>MM_WIM_OPEN 表示波形输入设备开启成功<br>MM_WIM_DATA 表示一个缓冲区已满。<br>此时dwParam1中携带有数据缓冲区头结的指针。正是通过这个指针,才可以随机地访问缓冲区中的每一个采样数据。如下面程序所示:<br><br><br>…<br>procedure TRecorder.CallBack(uMsg,dwInstance,dwParam1,dwParam2 : DWORD); <br>stdcall;<br>var i:Integer;<br>SPByte : ^Byte; //假设在打开设备时采用8位量化<br>SingleData : Integer;<br>BEGIN<br>case uMsg of //uMsg是Windows的消息标记号<br>MM_WIM_OPEN : //波形输入设备开启成功发回的消息<br>…<br>MM_WIM_DATA : //一个缓冲区已满发回的消息<br>begin<br>SPByte := Pointer(dwParam1);<br>for i :=0 to Recorder.DataLength-1 do<br>begin<br>SingleData := SPByte^; //通过SPByte来访问缓冲区中的数据<br>…<br>Inc(SPByte);<br>end;<br>end;<br>MM_WIM_CLOSE : //波形输入设备关闭成功发回的消息<br>…<br>end;<br>END;<br>…<br><br>MM_WIM_CLOSE 表示波形输入设备关闭成功。当波形输入设备关闭后,别忘了用Wave InPrepareHeader和GlobalFreePtr来释放缓冲区内存。<br><br>4. 关闭语音输入设备<br><br>waveInStop(hWaveIn) 停止语音输入<br>waveInReset(hWaveIn) 重置语音输入设备<br>waveInClose(hWaveIn) 关闭语音输入设备。其中hWaveIn是WaveInOpen得到的设备句柄。<br>在关闭语音输入设备前,必须重置语音输入设备,否则系统会出现这样的错误提示: &quot;MMSYSTEM033 媒体数据仍在播放中,请重置设备或等到数据播放完毕&quot;。但是只有当一个缓冲区填满数据后,才能重置语音输入设备<br>以上波形输入函数,若调用成功则返回0;否则返回非0,此时可以用waveInGetErrorText函数来得到出错信息,这样做的目的是方便调试。<br><br><br>三、必须注意的几点<br><br>以上阐述了作为A/D卡的声卡编程技术,但是还必须注意以下几点<br><br><br>1. 声卡的采样频率并不只限于11025Hz,22050Hz,44100 Hz三种,大多数声卡的采样频率在一定的范围内是可调的(当然会存在一定的偏差)。有的声卡的最高采样频率可达200K Hz(有可能随不同品牌而异)。<br>2. 缓冲区不能设得太小,否则也会造成采样的不连续。在作者的声卡上,若采用16为量化,22050Hz的采样速率,缓冲区设为1K字节,理论上每秒钟可以得到22050*2个字节的数据,实际上每秒钟只能得到大约16000*2个字节的数据。若缓冲区设为2K字节,则与理论值一致。<br>3. A/D转化后的数据格式是PCM格式,即:若是8位量化,对应着8位无符号数据,0对应着负满幅值,128对应着零电平,255对应着正满幅值;若是16位量化,对应着16位有符号数据,-32768对应着负满幅值,0对应着零电平,32767对应着正满幅值。编程过程中应注意所声明的数据类型是否与之相符合,比如在Delphi4.0中,8位无符号数据对应着 Byte型数据,16位有符号数据对应着SmallInt型数据。<br>4. 由于声卡的输入端往往带有隔直电容,所以不能用声卡直接对直流量进行采集。解决的办法就是将这个隔直电容短接。<br>5. 同样地,利用windows的API函数和声卡的D/A功能也可以使声卡产生模拟音频信号输出。
 
waveOutGetPitch原型<br>MMRESULT waveOutGetPitch(<br> &nbsp;HWAVEOUT hwo, &nbsp; &nbsp; //Handle of the waveform-audio output device. <br> &nbsp;LPDWORD pdwPitch &nbsp;<br>);<br> <br>控制音量要用到waveOutSetVlume函数,此函数的声明在MMSystem.h单元中。<br>此单放在Delphi目录下的Source/rtl/win中<br><br>主意这儿用到两个函数waveOutSetVlume和waveOutGetVolume,不要用混,<br>我第一次用它们时就给搞混了,好费事。<br><br>程序如下:<br><br>三个TrackBar的属性设置:Max:=255;Frequency:=16;<br><br>unit Unit1;<br><br>interface<br><br>uses<br> &nbsp;Windows, Messages, SysUtils, Classes, Graphics, Controls, Forms, Dialogs,<br> &nbsp;ComCtrls, mmsystem;<br><br>type<br> &nbsp;TForm1 = class(TForm)<br> &nbsp; &nbsp;TrackBar1: TTrackBar;<br> &nbsp; &nbsp;TrackBar2: TTrackBar;<br> &nbsp; &nbsp;TrackBar3: TTrackBar;<br> &nbsp; &nbsp;procedure TrackBar1Change(Sender: TObject);<br> &nbsp; &nbsp;procedure TrackBar2Change(Sender: TObject);<br> &nbsp; &nbsp;procedure TrackBar3Change(Sender: TObject);<br> &nbsp; &nbsp;procedure FormCreate(Sender: TObject);<br> &nbsp;private<br> &nbsp; &nbsp;{ Private declarations }<br> &nbsp;public<br> &nbsp; &nbsp;{ Public declarations }<br> &nbsp;end;<br><br>var<br> &nbsp;Form1: TForm1;<br><br>implementation<br><br>{$R *.DFM}<br><br>//音量控制<br>procedure TForm1.TrackBar1Change(Sender: TObject);<br>var<br>t,v:Longint;<br>begin<br>// if a=0 then Exit;<br> t:=TrackBar1.Position;<br> v:=(t shl 8)or(t shl 24);<br> waveOutSetVolume(0,v);<br>end;<br><br>//左声道控制<br>procedure TForm1.TrackBar2Change(Sender: TObject);<br>var<br>t,v:Longint;<br>begin<br> &nbsp;t:=TrackBar2.Position;<br> &nbsp;waveOutGetVolume(0,@v);<br> &nbsp;v:=v and $ffff0000 or (t shl 8);<br> &nbsp;waveOutSetVolume(0,v);<br>end;<br><br>//右声道控制<br>procedure TForm1.TrackBar3Change(Sender: TObject);<br>var<br>t,v:Longint;<br>begin<br> &nbsp;t:=TrackBar3.Position;<br> &nbsp;waveOutGetVolume(0,@v);<br> &nbsp;v:=v and $0000ffff or (t shl 24);<br> &nbsp;waveOutSetVolume(0,v);<br>end;<br><br>//获得当前音量值<br>procedure TForm1.FormCreate(Sender: TObject);<br>var<br>v:longint;<br>begin<br> &nbsp;waveOutGetVolume(0,@v);<br> &nbsp;TrackBar2.Position:=hi(v);<br> &nbsp;TrackBar3.Position:=hi(v shr 16);<br> &nbsp;if hi(v)&gt;hi(v shr 16) then<br> &nbsp; TrackBar1.Position:=hi(v)<br> &nbsp;else<br> &nbsp; TrackBar1.Position:=hi(v shr 16);<br>end;<br><br>end.<br><br><br>函数说明:<br><br>waveOutSetVolume<br>第一个参数是波形文件输出设备标识符;<br>第二个参娄是音量大小。这是一个32位的整数,低16位表示左声道的音量,<br>高16位表示右声道的音量。<br><br>waveOutGetVolume<br>第一个参数是波形文件输出设备标识符;<br>第二个参数是一个32位整数的指针。
 
多谢以上二位朋友的回答,但你们好像没看清我提出的问题的“内容”,所以没为送!<br>如果你们要解释给我听的,只要解释为何没有获取“WaveIn”当前的音量的API或方法!
 
来自:newditter, 时间:2005-5-26 4:36:48, ID:3084364 <br>多谢以上二位朋友的回答,但你们好像没看清我提出的问题的“内容”,所以没为送!<br>如果你们要解释给我听的,只要解释为何没有获取“WaveIn”当前的音量的API或方法! <br> <br><br>这个问题可能根本不要回答。(不知道我有没有正确理解楼主的意思)<br><br>可能楼主对音响方面不太熟悉。作为信号的输入后,借由声卡运放与功放电路进行放大然后输出,以驱动音箱发声,所以我们有WaveOutVol,如果输入功率不够的话那么外置音箱还需要进行信号放大,相当于硬件WaveOutVol。而放大是连有效信号和噪声一起放大的,所以我们在信号的获得上最好是得到最原始干净的信号。如果要有WaveIn的音量控制,那也是上级输入设备的事情,而不是声卡的事情。就算声卡有WaveIn的音量控制那么电路实现上也是放大电路,使用输出电路就可以了。更何况在电声工程上是严禁对输入进行干涉的,因为会加入不必要得噪声从而影响信号的纯净度。 所以任何声卡都不会对输入信号进行过多干预,它仅仅是个接受器,不但没有放大,它还会加上负反馈来防止你放大(消除电路本底噪声)。<br>所以,系统API也不会提供这个从根本上就不会存在的控制,所以,永远不会有原生的所谓的WaveIn的音量控制,只要涉及到音量控制必然是WaveOut的控制,因为这部分一定是由声卡输入的后一级电路完成。
 
呵呵,老兄只剩十几分了,不够用什么加啊?<br>你的 reditter、ditter 帐户和 newditter 导来导去有意思么?<br>http://www.delphibbs.com/delphibbs/listq.asp?sort=6&amp;type=2&amp;expert=newditter
 
我还有十多个用户!因为想出多的分,所以注册多几个用户!<br>我是积分差不多有8000分,你信吗?
 
帮你顶顶吧
 
分不是万能的,你给100000万分,让我开发一个windows操作系统;<br>我狂跳楼......
 

Similar threads

S
回复
0
查看
3K
SUNSTONE的Delphi笔记
S
S
回复
0
查看
2K
SUNSTONE的Delphi笔记
S
I
回复
0
查看
681
import
I
S
回复
0
查看
939
SUNSTONE的Delphi笔记
S
后退
顶部