在QTimer源码分析(以Windows下实现为例) 一文中,我们看到了Qt在windows下对计时器的使用:
对于间隔为零的情况,Qt并没有动用系统的计时器
对于间隔非零的情况
间隔小于20ms 且系统支持多媒体计时器,则使用多媒体计时器
否则,使用普通计时器
Qt 的这种策略应该能很好地满足我们的需求了,但qtcn上一个网友还是比较期待自己直接调用系统的多媒体计时器。既然这样,自己还是尝试写写吧,写一个自己的Timer类
代码
代码还是比较简单的,头文件 mmtimer.h 如下:
#ifndef MMTIMER_H
#define MMTIMER_H
#include <qt_windows.h>
#include <QtCore/QObject>class MMTimer : public QObject
{
Q_OBJECT
public:
explicit MMTimer(int interval, QObject *parent = 0);
~MMTimer();signals:
void timeout();public slots:
void start();
void stop();friend void WINAPI CALLBACK mmtimer_proc(uint, uint, DWORD_PTR, DWORD_PTR, DWORD_PTR);
private:
int m_interval;
int m_id;
};#endif // MMTIMER_H
源码文件 mmtimer.cpp 如下:
#include "mmtimer.h"
#include <MMSystem.h>
#ifdef __MINGW32__ //w32api bug
#define TIME_KILL_SYNCHRONOUS 0x0100
#endif
void WINAPI CALLBACK mmtimer_proc(uint timerId, uint, DWORD_PTR user, DWORD_PTR, DWORD_PTR)
{
MMTimer *t = reinterpret_cast<MMTimer*>(user);
emit t->timeout();
}MMTimer::MMTimer(int interval, QObject *parent) :
QObject(parent),m_interval(interval),m_id(0)
{
}MMTimer::~MMTimer()
{
stop();
}void MMTimer::start()
{
m_id = timeSetEvent(m_interval, 1, mmtimer_proc, (DWORD_PTR)this,
TIME_CALLBACK_FUNCTION | TIME_PERIODIC | TIME_KILL_SYNCHRONOUS);}
void MMTimer::stop()
{
if (m_id){
timeKillEvent(m_id);
m_id = 0;
}
}
说明
上面的代码应该不需要什么解释了:
timeSetEvent 和 timeKillEvent 可直接查阅 MSDN
另外,MinGW的win32api包,对TIME_KILL_SYNCHRONOUS没有定义,代码中做了一点修正
请确保正确链接所需要的库
LIBS += -lwinmm
注意:MSDN 对timeSetEvent的介绍中这么说的(对此不做评论)
Note This function is obsolete. New applications should use CreateTimerQueueTimer to create a timer-queue timer.
作者:dbzhang800