分享

VC++ 高精度定时timeSetEvent和回调成员方法

 昵称6004610 2013-01-08
VC++ 高精度定时timeSetEvent和回调成员方法

 

Platform SDK: Windows Multimedia中的timeSetEvent函数和timeKillEvent函数可以启动和停止一个精度达到1ms的高精度定时器。

原型:

MMRESULT timeSetEvent(

  UINT           uDelay,     

  UINT           uResolution,

  LPTIMECALLBACK lpTimeProc, 

  DWORD_PTR      dwUser,     

  UINT           fuEvent     

);

Parameters

uDelay

Event delay, in milliseconds. If this value is not in the range of the minimum and maximum event delays supported by the timer, the function returns an error.

事件延迟,毫秒。如果这个值不在定时器支持的最小和最大延时之间的话,函数将返回一个错误。

uResolution

Resolution of the timer event, in milliseconds. The resolution increases with smaller values; a resolution of 0 indicates periodic events should occur with the greatest possible accuracy. To reduce system overhead, however, you should use the maximum value appropriate for your application.

定时器的分辨力,毫秒。小的数值表示高的分辨力;分辨力为0表示周期事件应该以允许的最高精度发生。尽管如此为减小系统开销,你应该使用适合你应用的最大值。

lpTimeProc

Pointer to a callback function that is called once upon expiration of a single event or periodically upon expiration of periodic events. If fuEvent specifies the TIME_CALLBACK_EVENT_SET or TIME_CALLBACK_EVENT_PULSE flag, then the lpTimeProc parameter is interpreted as a handle to an event object. The event will be set or pulsed upon completion of a single event or periodically upon completion of periodic events. For any other value of fuEvent, the lpTimeProc parameter is interpreted as a function pointer with the following signature: void (CALLBACK)(UINT uTimerID, UINT uMsg, DWORD_PTR dwUser, DWORD_PTR dw1, DWORD_PTR dw2);

指向回调函数的指针。发生单次事件时它被调用一次,发生周期事件时,它被周期调用。如果fuEventTIME_CALLBACK_EVENT_SETTIME_CALLBACK_EVENT_PULSE标识,这时lpTimeProc被翻译成事件的一个句柄。根据单个事件或周期事件的完成,事件会被设置或触发。对于fuEvent的其它值,lpTimeProc被翻译为一个函数指针,它具有下面的签名:void (CALLBACK)(UINT uTimerID, UINT uMsg, DWORD_PTR dwUser, DWORD_PTR dw1, DWORD_PTR dw2);

dwUser

User-supplied callback data.

用户提供的回调函数数据

fuEvent

Timer event type. This parameter may include one of the following values.

定时事件类型。这个参数可能包含以下值的一个:

Value

Meaning

TIME_ONESHOT

Event occurs once, after uDelay milliseconds.

事件在uDelay毫秒后发生一次

TIME_PERIODIC

Event occurs every uDelay milliseconds.

事件每隔uDelay毫秒发生一次

The fuEvent parameter may also include one of the following values.

fuEvent也包含下面的值

Value

Meaning

TIME_CALLBACK_FUNCTION

When the timer expires, Windows calls the function pointed to by the lpTimeProc parameter. This is the default.

当定时器超时,Windows调用lpTimeProc参数所指向的函数。这时默认的。

TIME_CALLBACK_EVENT_SET

When the timer expires, Windows calls the SetEvent function to set the event pointed to by the lpTimeProc parameter. The dwUser parameter is ignored.

当定时器超时,Windows调用SetEvent函数来设置lpTimeProc参数所指向的事件。dwUser参数被忽略。

TIME_CALLBACK_EVENT_PULSE

When the timer expires, Windows calls the PulseEvent function to pulse the event pointed to by the lpTimeProc parameter. The dwUser parameter is ignored.

当定时器超时,Windows调用PulseEvent函数来触发lpTimeProc参数所指向的事件。dwUser参数被忽略。

TIME_KILL_SYNCHRONOUS

Passing this flag prevents an event from occurring after the timeKillEvent() function is called.

传递这个标识将在调用timeKillEvent()后阻止事件发生。

Return Values

Returns an identifier for the timer event if successful or an error otherwise. This function returns NULL if it fails and the timer event was not created. (This identifier is also passed to the callback function.)

如果成功返回定时器事件的一个标识符,否则返回一个错误。如果它失败了,这个函数返回NULL,定时器事件不会被创建。(这个标识符同样会被传递给回调函数)。

Remarks

Each call to timeSetEvent for periodic timer events requires a corresponding call to the timeKillEvent function.

Creating an event with the TIME_KILL_SYNCHRONOUS and the TIME_CALLBACK_FUNCTION flag prevents the event from occurring after the timeKillEvent function is called.

每个对timeSetEvent的周期性定时调用都需要一个相应的timeKillEvent函数的调用。

通过TIME_KILL_SYNCHRONOUS方式创建一个事件,TIME_CALLBACK_FUNCTION标识在timeKillEvent调用后将阻止事件的发生。

 

由于在VC++中,成员方法都有一个this指针,不能将成员方法直接作为回调函数使用。解决的方法是使用一个static的方法作为回调函数。但是如果仅仅这样的话,我们将不能使用实例中的成员变量和成员方法等。这意味着在很多情况下回调达不到我们的需求。参考网上的一些方法,注意到timeSetEvent有一个DWORD类型的用户提供的回调函数数据dwUser ,在此处如果将对象的this指针传入,然后在静态方法内通过this指针来调用成员方法。这样就可以间接的实现成员函数的回调了。这个方法是最简单的方法。但是它也存在明显的缺陷,那就是调用深度多了一级,可能引发执行效率的问题。网上也有很好的解决方法,但是会比较复杂。需要分析清成员函数调用的机制,甚至还需要编写一些汇编的代码,在这里就先不介绍了。

Example:

在类CTimerCallbackDlg

protected:

       virtual void OnMMTimer();//声明要回调的成员方法

//作为回调函数的静态方法

       static void CALLBACK TimerCallBack(UINT uTimerID, UINT uMsg, DWORD dwUser, DWORD dw1, DWORD dw2)

       {

              CTimerCallbackDlg * pThis=(CTimerCallbackDlg*)dwUser;//this指针获得实例的指针

              pThis->OnMMTimer();//调用要回调的成员方法

       }

/***********************************/

//一个定时器的使能禁能按钮

void CTimerCallbackDlg::OnTimerEn()

{

       // TODO: Add your control notification handler code here

       if(!m_en)

       {

              //返回定时器的id,用于停止定时器,回调静态函数,传入this指针

m_timerId=timeSetEvent(100,100,&TimerCallBack,(DWORD)this,1);

              m_en=true;

              SetDlgItemText(IDC_TimerEn,"Stop");

       }

       else

       {

              //关闭定时器

              timeKillEvent(m_timerId);

              m_en=false;

              SetDlgItemText(IDC_TimerEn,"Start");

       }

}

 

    本站是提供个人知识管理的网络存储空间,所有内容均由用户发布,不代表本站观点。请注意甄别内容中的联系方式、诱导购买等信息,谨防诈骗。如发现有害或侵权内容,请点击一键举报。
    转藏 分享 献花(0

    0条评论

    发表

    请遵守用户 评论公约

    类似文章 更多