Преобразование указателя функции-члена в TIMERPROC - PullRequest
7 голосов
/ 24 июня 2011

Как преобразовать указатель на функцию-член в тип TIMERPROC для использования с WINAPI SetTimer? Приведенный ниже фрагмент кода показывает, как я это делаю сейчас, но при компиляции я получаю эту ошибку:

ошибка C2664: «SetTimer»: невозможно преобразовать параметр 4 из «void (__stdcall CBuildAndSend :: *) (HWND, UINT, UINT_PTR, DWORD)» в «TIMERPROC»

Обратный вызов должен быть привязан к исходному экземпляру класса. Если есть лучший способ сделать это, я весь слух. Спасибо.

class CMyClass
{
public:
    void (CALLBACK CBuildAndSend::*TimerCbfn)( HWND hwnd, UINT uMsg, UINT_PTR idEvent, DWORD dwTime );

private:
    void CALLBACK TimeoutTimerProc( HWND hwnd, UINT uMsg, UINT_PTR idEvent, DWORD dwTime );
};

CMyClass::CMyClass()
{
    ...

    this->TimerCbfn = &CBuildAndSend::TimeoutTimerProc;

    ...

    ::CreateThread(
        NULL,                           // no security attributes
        0,                              // use default initial stack size
        reinterpret_cast<LPTHREAD_START_ROUTINE>(BasThreadFn), // function to execute in new thread
        this,                           // thread parameters
        0,                              // use default creation settings
        NULL                            // thread ID is not needed
        )
}

void CALLBACK CMyClass::TimeoutTimerProc( HWND hwnd, UINT uMsg, UINT_PTR idEvent, DWORD dwTime )
{
    ...
}

static DWORD MyThreadFn( LPVOID pParam )
{
    CMyClass * pMyClass = (CMyClass *)pParam;

    ...

    ::SetTimer( NULL, 0, BAS_DEFAULT_TIMEOUT, pMyClass->TimerCbfn ); // <-- Error Here

    ...
}

1 Ответ

9 голосов
/ 24 июня 2011

Member-function и TIMEPROC не являются совместимыми типами.

Вам нужно сделать функцию-член static. Затем он будет работать, предполагая, что список параметров одинаков как в статической функции-члене, так и в TIMEPROC.

class CMyClass
{
public:
    //modified
    void (CALLBACK *TimerCbfn)(HWND hwnd, UINT uMsg, UINT_PTR idEvent, DWORD dwTime);

private:
    //modified
    static void CALLBACK TimeoutTimerProc(HWND hwnd, UINT uMsg, UINT_PTR idEvent, DWORD dwTime );
};

Указатель на функцию, а также функция-член модифицированы. Теперь это должно работать.

Теперь, поскольку функция обратного вызова стала статической, она не может получить доступ к нестатическим членам класса, потому что у вас нет указателя this в функции.

Чтобы получить доступ к нестатическим членам, вы можете сделать это:

class CMyClass
{
public:

    static void CALLBACK TimerProc( HWND hwnd, UINT uMsg, UINT_PTR idEvent, DWORD dwTime );

    //add this static member
    static std::map<UINT_PTR, CMyClass*> m_CMyClassMap; //declaration
};

//this should go in the  CMyClass.cpp file
std::map<UINT_PTR, CMyClass*> CMyClass::m_CMyClassMap;  //definition

static DWORD MyThreadFn( LPVOID pParam )
{
    CMyClass * pMyClass = (CMyClass *)pParam;

    UINT_PTR id = ::SetTimer( NULL, 0, BAS_DEFAULT_TIMEOUT, CMyClass::TimerProc);

    //store the class instance with the id as key!        
    m_CMyClassMap[id]= pMyClass; 
}

void CALLBACK CMyClass::TimerProc( HWND hwnd, UINT uMsg, UINT_PTR idEvent, DWORD dwTime )
{
    //retrieve the class instance
    CMyClass *pMyClass= m_CMyClassMap[idEvent];

    /*
      now using pMyClass, you can access the non-static 
      members of the class. e.g
      pMyClass->NonStaticMemberFunction();
    */
}

Я удалил TimerCbfn из своей реализации, так как он на самом деле не нужен. Вы можете передать TimerProc напрямую SetTimer как последний аргумент.

...