用C++将回调函数转换为DWORD_PTR

Convert Callback Function to DWORD_PTR in C++

本文关键字:DWORD PTR 转换 函数 C++ 回调      更新时间:2023-10-16

实际上是c++的新手。我写了这个代码,但说无法从"waveOutProc"转换为"DWORD_PTR"。你能告诉我如何修复它吗?

感谢

void CALLBACK Audio::waveOutProc(HWAVEOUT hWaveOut, unsigned int uMsg,
unsigned long dwInstance, unsigned long dwParam1,
unsigned long dwParam2)
{
/*
* pointer to free block counter
*/
int* freeBlockCounter = (int*)dwInstance;
/*
* ignore calls that occur due to openining and closing the
* device.
*/
if(uMsg != WOM_DONE) { return ; }
EnterCriticalSection(&waveCriticalSection) ;
(*freeBlockCounter)++ ;
LeaveCriticalSection(&waveCriticalSection) ;
}

//////////////////////////////////////////////////////////////////////////////////

void Audio::playSound(const char* filename)
{
HWAVEOUT hWaveOut ;
HANDLE hFile;
WAVEFORMATEX wfx ;
char buffer[1024];
int i;
...
if(waveOutOpen(
&hWaveOut,
WAVE_MAPPER,
&wfx,
(DWORD_PTR)waveOutProc,            ///////////Error Point
(DWORD_PTR)&waveFreeBlockCount,
CALLBACK_FUNCTION
) != MMSYSERR_NOERROR) {
fprintf(stderr, "unable to open wave mapper devicen");
ExitProcess(1);
}
...
}
将函数指针转换为数据指针是一种未定义的行为,因此首先不应该这样做。(我知道win-api函数正期待着这一点)。

此外,除非处理隐式this参数,否则不能在C/C++中将成员函数作为回调传递。

您的目标回调具有以下签名

void CALLBACK waveOutProc(
HWAVEOUT hwo,
UINT uMsg,
DWORD_PTR dwInstance,
DWORD_PTR dwParam1,
DWORD_PTR dwParam2
);

而Audio::waveOutProc可能是一个成员函数,它隐含着这个论点。

void CALLBACK waveOutProc(Audio*,
HWAVEOUT hwo,
UINT uMsg,
DWORD_PTR dwInstance,
DWORD_PTR dwParam1,
DWORD_PTR dwParam2
);

只需将waveOutProc定义为静态函数或自由函数即可。

waveOutProc是一个函数,因此不能将其强制转换为积分类型(值)。此外,我想waveOutProc可能也不是你班的静态成员?只能将静态函数作为回调函数传递。

您在评论中发布的错误表明waveOutProcAudio类的成员函数,C++不允许您将成员函数分配给需要"正常"函数的参数或变量。这是因为成员函数有一个名为this的隐式参数,它是指向Audio类实例的指针。

相反,编写一个所谓的静态成员函数(static关键字表示没有隐含的this参数)是一种很好的做法,它封装了要调用的成员函数。这是可能的,因为waveOutOpen将用户数据变量作为它的第五个参数,然后将其传递给静态回调。静态成员函数包装器比只使回调为静态要好,因为这样您就可以访问所有类成员变量(而不是仅访问一个变量,例如freeBlockCounter)。您的静态成员函数包装器可能如下所示:

class Audio {
private:
int freeBlockCounter;
public:
....
static void CALLBACK waveOutProcWrapper(HWAVEOUT hWaveOut, unsigned int uMsg,
unsigned long dwInstance,
unsigned long dwParam1,
unsigned long dwParam2);
void waveOutProc(HWAVEOUT hWaveOut, unsigned int uMsg, unsigned long dwParam1,
unsigned long dwParam2);
};

以及wrapperProc:的实现

void CALLBACK Audio::waveOutProcWrapper(HWAVEOUT hWaveOut, unsigned int uMsg,
unsigned long dwInstance,
unsigned long dwParam1,
unsigned long dwParam2) {
((Audio*)dwInstance)->waveOutProc(hWaveOut, uMsg, dwParam1, dwParam2);
}

请注意dwInstance参数是如何"转换"为隐式this参数的。您现在可以通过以下方式提供waveOutProcWrapperwaveOutOpenint:

if(waveOutOpen(
&hWaveOut,
WAVE_MAPPER,
&wfx,
(DWORD_PTR)waveOutProcWrapper,            ///////////Error Point
(DWORD_PTR)this,
CALLBACK_FUNCTION) != MMSYSERR_NOERROR) {
fprintf(stderr, "unable to open wave mapper devicen");
ExitProcess(1);
}