在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.