在VC编程中,用SetTimer可以定义一个定时器,到时间了,就响应OnTimer消息,但这种定时器精度太低了。如果需要精度更高一些的定时器(精 确到1ms),可以使用下面的高精度多媒体定时器进行代码优化,可以达到毫秒级的精度,而且使用方便。先要包含头文件"mmsystem.h"和库文 件"winmm.lib"。

虽然Win95下可视化开发工具如VC、Delphi、C++   Builder等都有专用的定时器控件Timer,而且使用很方便,可以实现一定的定时功能,但最小计时精度仅为55ms,且定时器消息在多任务操作系统 中的优先级很低,不能得到及时响应,往往不能满足实时控制环境下的应用。不过Microsoft公司在Win32   API函数库中已经为用户提供了一组用于高精度计时的底层函数,如果用户使用得当,计时精度可到1ms。这个计时精度、对于一般的实时系统控制完全可以满足要求。现将由C++   Builder   4.0提供的重新封装后的一组与时间相关的主要接口函数(函数名、参数、功能与Win32   API基本相同)说明如下:

1.DWORD   timeGetTime(void)

  返回从Windows启动开始经过的毫秒数。最大值为232,约49.71天。

2.MMRESULT   timeSetEvent(

  UINT   uDelay,

  UINT   uResolution,

  LPTIMECALLBACK   lpTimeProc,

  DWORD   dwUser,

  UINT   fuEvent)

  该函数设置一个定时回调事件,此事件可以是一个一次性事件或周期性事件。事件一旦被激活,便调用指定的回调函数,成功后返回事件的标识符代码,否则返回NULL。参数说明如下:

uDelay:以毫秒指定事件的周期。

  UResolution:以毫秒指定延时的精度,数值越小定时器事件分辨率越高。缺省值为1ms。

  LpTimeProc:指向一个回调函数。

  DwUser:存放用户提供的回调数据。

  FuEvent:指定定时器事件类型:

  TIME_ONESHOT:uDelay毫秒后只产生一次事件

  TIME_PERIODIC   :每隔uDelay毫秒周期性地产生事件。

3.MMRESULT   timeKillEvent(UINT   uTimerID)

  该函数取消一个指定的定时器回调事件。uTimerID标识要取消的事件(由timeSetEvent函数返回的标识符)。如果成功则返回TIMERR_NOERROR,如果定时器时间不存在则返回MMSYSERR_INVALPARAM。

4.回调函数

  void   CALLBACK   TimeProc(

  UINT   uID,

  UINT   uMsg,

  DWORD   dwUser,

  DWORD   dw1,

  DWORD   dw2);

该函数是一个应用程序定义的回调函数,出现定时器事件时该函数被调用。TimeProc是应用程序定义的函数名的占位符。使用该函数

时要注意的是,它只能调用以下有限的几组API函数:PostMessage,timeGetSystemTime,  timeGetTime,   timeSetEvent,timeKillEvent

,midiOutShortMsg,   midiOutLongMsg,OutputDebugString。同时也不要使用完成时间很长的API函数,程序尽可能简短。

  使用以上一组函数就可以完成毫秒级精度的计时和控制(在C++Builder中使用时要将头文件mmsystem.h加到程序中)。由于将定时控

制精确到几毫秒,定时器事件将占用大量的CPU时间和系统资源,所以在满足控制要求的前提下,应尽量将参数uResolution的数值增大。而

且定时器实时控制功能完成后要尽快释放。

注意以下几点问题:

一、回调函数的参数不能有误,否则可能引起程序崩掉;

二、事件调用周期uDelay不能小于事件处理时间,否则会引起程序崩溃;

三、通过dwUser给回调函数传递参数

例程如下:

MMRESULT g_wTimerID = 0;

//回调函数,参数不能有错

void CALLBACK CDsisiiDlg::SendFun(UINT wTimerID, UINT msg, DWORD dwUser, DWORD dwl, DWORD dw2)

{

CDsisiiDlg* pdcpackerdlg = (CDsisiiDlg*)dwUser;

...

}

bool  CDsisiiDlg::CreateTimer()

{

TIMECAPS   tc;

UINT wTimerRes;

//设置多媒体定时器

if(timeGetDevCaps(&tc,sizeof(TIMECAPS))!=TIMERR_NOERROR)//向机器申请一个多媒体定时器

return false;

//获得机器允许的时间间隔(一般可达到1毫秒)

wTimerRes=min(max(tc.wPeriodMin,1),tc.wPeriodMax);

//定时器开始工作

timeBeginPeriod(wTimerRes);

//每过6毫秒调用回调函数timerback(),wTimerID为定时器ID.TIME_PERIODIC表周期性调用,TIME_ONESHOT表只产生一次事件

g_wTimerID = timeSetEvent(6,  wTimerRes, (LPTIMECALLBACK)SendFun,  (DWORD)this, TIME_PERIODIC);

if(g_wTimerID == 0)

return false;

return true;

}

//删除定时器

void CDsisiiDlg::DestroyTimer()

{

if (g_wTimerID)

{

timeKillEvent(g_wTimerID);

g_wTimerID = 0;

}

}

一下为在QT下使用windows多媒体计时器

在QTimer源码分析(以Windows下实现为例) 一文中,我们看到了Qt在windows下对计时器的使用:

对于间隔为零的情况,Qt并没有动用系统的计时器

对于间隔非零的情况

间隔小于20ms 且系统支持多媒体计时器,则使用多媒体计时器

否则,使用普通计时器

Qt 的这种策略应该能很好地满足我们的需求了,但qtcn上一个网友还是比较期待自己直接调用系统的多媒体计时器。既然这样,自己还是尝试写写吧,写一个自己的Timer类

代码

代码还是比较简单的,头文件 mmtimer.h 如下:

#ifndef MMTIMER_H

#define MMTIMER_H

#include

#include

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

#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(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.

http://blog.sina.com.cn/s/blog_668aae780101dfij.html

VC++或QT下 高精度 多媒体定时器的更多相关文章

  1. Windows下用VC与QT编译MPI程序入门

    MPI是信息传递接口的简称,常用来进行进程间.机器间的通信与并行计算.一般而言,MPI都会部署在*nix系统下,在Windows下面直接编译.配置MPI并不容易.本文利用MS提供的编译好的MPI的版本 ...

  2. C#中自定义高精度Timer定时器的实例教程

    Timer 用于以用户定义的事件间隔触发事件.Windows 计时器是为单线程环境设计的,其中,UI 线程用于执行处理.它要求用户代码有一个可用的 UI 消息泵,而且总是在同一个线程中操作,或者将调用 ...

  3. (笔记)CanOpen协议【CanFestival】移植方法 支持VC、QT、STM32

    转自http://bbs.21ic.com/icview-878522-1-1.html   前段时间学习了CanOpen协议,到网上下载的CanFestival3-10源码,移植到VC.QT.STM ...

  4. CanOpen协议【CanFestival】移植方法 支持VC、QT、STM32

    前段时间学习了CanOpen协议,到网上下载的CanFestival3-10源码,移植到VC.QT.STM32等平台,由于网上的资源较少,走了不少弯路,移植好使用过程中才逐渐暴露出各种问题,比如OD字 ...

  5. 【转】Qt下使用glut库

    ps:这个说的很明白,尤其是win10环境下用mingw环境时编程时碰到的问题, 1.加 windows.h 2.在.pro 添加libs     博文地址:Qt下使用glut库   本人使用的环境 ...

  6. Qt下libusb-win32的使用方法(转)

    源:Qt下libusb-win32的使用方法 之前一直找不到适合WIN7下的Tiny6410的USB下载软件,正好这几天开始学习USB,所以打算自己写一个专门用于Tiny6410的WIN7下的USB下 ...

  7. Qt下libusb-win32的使用方法

    之前一直找不到适合WIN7下的Tiny6410的USB下载软件,正好这几天开始学习USB,所以打算自己写一个专门用于Tiny6410的WIN7下的USB下载软件. 发现了libusb这个库可以用作无驱 ...

  8. Qt下QString转char*

    Qt下面,字符串都用QString,确实给开发者提供了方便,想想VC里面定义的各种变量类型,而且函数参数类型五花八门,经常需要今年新那个类型转换 Qt再使用第三方开源库时,由于库的类型基本上都是标准的 ...

  9. Qt下 QString转char*(转)

    Qt下面,字符串都用QString,确实给开发者提供了方便,想想VC里面定义的各种变量类型,而且函数参数类型五花八门,经常需要今年新那个类型转换 Qt再使用第三方开源库时,由于库的类型基本上都是标准的 ...

随机推荐

  1. 学习C++语言的50条忠告

    50条忠告:(其中有几条觉得写的不够贴切,所以删了,发了余下的部分) 1.把C++当成一门新的语言学习: 2.看<Thinking In C++>,不要看<C++变成死相>: ...

  2. 从零开始PHP学习 - 第四天

    写这个系列文章主要是为了督促自己  每天定时 定量消化一些知识! 同时也为了让需要的人 学到点啥~! 本人技术实在不高!本文中可能会有错误!希望大家发现后能提醒一下我和大家! 偷偷说下 本教程最后的目 ...

  3. [原创]抢先DriverStudio夺取机器控制权(下篇)

    原文链接:抢先DriverStudio夺取机器控制权(下篇) 上篇仅仅说到如何抢先DriverStudio,并在结尾留给大家一个遐想.现在我进一步拓展这个遐想,从而给大家更多的遐想.: ) 那么现在我 ...

  4. 解决Oracle 11gR2 空闲连接过多,导致连接数满的问题

    今天又遇到了11gR2连接数满的问题,以前也遇到过,因为应用那边没有深入检查,没有找到具体原因,暂且认为是这个版本Oracle的BUG吧. 上次的处理办法是用Shell脚本定时在系统中kill  v$ ...

  5. delphi 7中使用idhttp抓取网页 解决假死现象

    在delphi 7中使用idhttp抓取网页,造成窗口无反应的假死状态.通过搜索获得两种方法. 1.写在线程中,但是调用比较麻烦 2.使用delphi 提供的idantifreeze(必须安装indy ...

  6. WPF的重要新概念

    原文 http://www.cnblogs.com/free722/archive/2011/11/12/2238654.html 逻辑树与可视树 XAML天生就是用来呈现用户界面的,这是由于它具有层 ...

  7. 一个基于Qt的截屏程序

    最近有一个arm板上的程序需要重写用户手册,在网上找了许久,没找到合适的截屏工具.于是只好自己动手做一个了. 因为arm板上有已经有了Qt环境,于是想到用 Qt的QPixmap::grabWindow ...

  8. python setattr(),getattr()函数

    setattr(object,name,value): 作用:设置object的名称为name(type:string)的属性的属性值为value,属性name可以是已存在属性也可以是新属性. get ...

  9. 指定html内容下载为word文档

    解决思路是:获取html内容并传到后台,后台把html内容转换为输入流再传给浏览器,浏览器直接下载 1.获取html内容并传到后台 $("#zxjdck .ad-xzzy-anniu&quo ...

  10. BZOJ 2716 Violet 3 天使玩偶 CDQ分治

    题目大意:初始给定平面上的一个点集.提供两种操作: 1.将一个点增加点集 2.查询距离一个点最小的曼哈顿距离 K-D树是啥...不会写... 我仅仅会CDQ分治 对于一个询问,查询的点与这个点的位置关 ...