原文链接地址:http://www.cnblogs.com/kangwang1988/archive/2010/09/16/1827872.html

微软公司在其多媒体Windows中提供了精确定时器的底层API支持。利用多媒体定时器可以很精确地读出系统的当前时间,并且能在非常精确的时间间隔内完成一个事件、函数或过程的调用。利用多媒体定时器的基本功能,可以通过两种方法实现精确定时。

1)使用timeGetTime()函数,该函数定时精度为ms级,返回从Windows启动开始所经过的时间。由于使用该函数是通过查询的方式进行定时控制的,所以,应该建立定时循环来进行定时事件的控制。

2)使用timeSetEvent()函数,该函数原型如下:

MMRESULT timeSetEvent(  UINT           uDelay,
UINT uResolution,
LPTIMECALLBACK lpTimeProc,
DWORD_PTR dwUser,
UINT fuEvent );

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

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

LpTimeProc:指向一个回调函数。

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

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

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

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

该函数的参数说明如下:参数uDelay表示延迟时间;参数uResolution表示时间精度,在Windows中缺省值为1ms;lpTimeProc表示回调函数,为用户自定义函数,定时调用; 参数dwUser表示用户提供的回调数据;参数fuEvent为定时器的事件类型,TIME_ONESHOT表示执行一次;TIME_PERIODIC:周期性执行。具体应用时,可以通过调用timeSetEvent()函数,将需要周期性执行的任务定义在lpTimeProc回调函数中(如:定时采样、控制等),从而完成所需处理的事件。需要注意的是:任务处理的时间不能大于周期间隔时间。另外,在定时器使用完毕后,应及时调用timeKillEvent()将之释放。下面这段代码的主要功能是设置两个时钟定时器,一个间隔是1ms,一个间隔是2s。每执行一次,把当前系统时钟值输入文件"cure.out"中,以比较该定时器的精确度。

# define ONE_MILLI_SECOND 1 //定义1ms和2s时钟间隔,以ms为单位 ;
# define TWO_SECOND 2000
# define TIMER_ACCURACY 1 //定义时钟分辨率,以ms为单位
UINT wTimerRes_1ms,wTimerRes_2s; //定义时间间隔
UINT wAccuracy; //定义分辨率
UINT TimerID_1ms,TimerID_2s; //定义定时器句柄
///////////////////////////////
CCureApp::CCureApp():fout("cure.out", ios::out) //打开输出文件"cure.out";
{
 // 给时间间隔变量赋值
 wTimerRes_1ms = ONE_MILLI_SECOND;
 wTimerRes_2s = TWO_SECOND;
 TIMECAPS tc;
 //利用函数timeGetDeVCaps取出系统分辨率的取值范围,如果无错则继续;
 if(timeGetDevCaps(&tc,sizeof(TIMECAPS))==TIMERR_NOERROR)
 {
   wAccuracy=min(max(tc.wPeriodMin, //分辨率的值不能超出系统的取值范围
    TIMER_ACCURACY),tc.wPeriodMax);
  //调用timeBeginPeriod函数设置定时器的分辨率
  timeBeginPeriod(wAccuracy);
  //设置定时器
  InitializeTimer();
 }
}
CCureApp:: ~CCureApp()
{
 fout <<"结束时钟"<< endl; //结束时钟
 timeKillEvent(TimerID_1ms); // 删除两个定时器
 timeKillEvent(TimerID_2s); // 删除设置的分辨率
 timeEndPeriod(wAccuracy);
}
void CCureApp::InitializeTimer()
{
 StartOneMilliSecondTimer();
 StartTwoSecondTimer();
}
//1ms定时器的回调函数,类似于中断处理程序,一定要声明为全局PASCAL函数,
//否则编译会有问题
void PASCAL OneMilliSecondProc(UINT wTimerID, UINT msg,DWORD dwUser,
DWORD dwl,DWORD dw2)
{
 // 定义计数器
 static int ms = 0;
 CCureApp *app = (CCureApp *)dwUser;
 // 取得系统时间,以ms为单位
 DWORD osBinaryTime = GetTickCount();
 //输出计数器值和当前系统时间
 app->fout<<++ms<<":1ms:"
}
// 加装1ms定时器
void CCureApp::StartOneMilliSecondTimer()
{
 if((TimerID_1ms = timeSetEvent(wTimerRes_1ms, wAccuracy,
  (LPTIMECALBACK) OneMil liSecondProc, // 回调函数;
  (DWORD)this, // 用户传送到回调函数的数据;
  TIME_PERIODIC)) == 0)//周期调用定时处理函数;
 {
  AfxMessageBox("不能进行定时!", MB_OK | MB_ICONASTERISK);
 }
 else
  fout << "16ms 计 时:" << endl; //不等于0表明加装成功,返回此定时器的句柄;
}

在精度要求较高的情况下,如要求定时误差不大于1ms时,还可以利用GetTickCount()函数返回自计算机启动后的时间,该函数的返回值是DWORD型,表示以ms为单位的计算机启动后经历的时间间隔。通过两次调用GetTickCount()函数,然后控制它们的差值来取得定时效果.下列的代码可以实现50ms的精确定时,其误差是毫秒级的。

// 起始值和中止值
DWORD dwStart, dwStop ;
dwStop = GetTickCount();
while(TRUE) {
 // 上一次的中止值变成新的起始值
 dwStart = dwStop ; // 此处添加相应控制语句
 do
 {
  dwStop = GetTickCount() ;
 }while(dwStop - 50 < dwStart) ;
}

用上述两种方式取得的定时效果虽然在许多场合已经满足实际的要求,但由于它们的精度只有毫秒级的,而且在要求定时时间间隔小时,实际定时误差大。对于精确度要求更高的定时操作,则应该使用QueryPerformanceFrequency()和QueryPerformanceCounter()函数。这两个函数是Visual C++提供并且仅供Windows 95及其后续版本使用,其精度与CPU的时钟频率有关,它们要求计算机从硬件上支持精确定时器。QueryPerformanceFrequency()函数和QueryPerformanceCounter()函数的原型如下:

BOOL QueryPerformanceFrequency (LARGE_INTEGER *lpFrequency);

BOOL QueryPerformanceCounter (LARGE_INTEGER *lpCount);

上述两个函数的参数的数据类型LARGE_INTEGER既可以是一个8字节长的整型数,也可以是两个4字节长的整型数的联合结构,其具体用法根据编译器是否支持64位而定。该类型的定义如下:

typedef union _LARGE_INTEGER

{

struct{

DWORD LowPart ; // 4字节整型数

LONG HighPart ; // 4字节整型数

};

LONG QuadPart ; // 8字节整型数

} LARGE_INTEGER ;

使用QueryPerformanceFrequency()和QueryPerformanceCounter()函数进行精确定时的步骤如下:

1、首先调用QueryPerformanceFrequency()函数取得高精度运行计数器的频率f,单位是每秒多少次(n/s),此数一般很大;

2、在需要定时的代码的两端分别调用QueryPerformanceCounter()以取得高精度运行计数器的数值n1、n2,两次数值的差值通过f换算成时间间隔,t=(n2-n1)/f,当t大于或等于定时时间长度时,启动定时器;

//////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////

自己的一个try

#include <stdio.h>
#include <stdlib.h>
#include <windows.h>
#pragma comment(lib,"Winmm.lib") # define ONE_MILLI_SECOND 1 //定义1ms和2s时钟间隔,以ms为单位 ;
# define TWO_SECOND 2000
# define TIMER_ACCURACY 1 //定义时钟分辨率,以ms为单位
void PASCAL OneMilliSecondProc(UINT wTimerID, UINT msg,DWORD dwUser,DWORD dwl,DWORD dw2)
{
printf("11111111111\n");
exit(0);
} void main() {
HANDLE hHandle; UINT wTimerRes_1ms,wTimerRes_2s;//定义时间间隔
UINT wAccuracy; //定义分辨率
UINT TimerID_1ms,TimerID_2s; //定义定时器句柄
wTimerRes_1ms = 5000;
if((TimerID_1ms = timeSetEvent(wTimerRes_1ms, wAccuracy,(LPTIMECALLBACK)OneMilliSecondProc, // 回调函数
(DWORD)(1), // 用户传送到回调函数的数据;
TIME_PERIODIC)) == 0)//周期调用定时处理函数
{
printf("start!!!!!!!!!!!\n");
}
else
{
printf("end!!!!!!!!!!!\n");
} while (1)
{
printf("hello!\n");
Sleep(1000);
}
}

timeSetEvent()函数的更多相关文章

  1. 时间的函数,sleep,clock,gettickcount,QueryPerformanceCounter(转)

    介绍 我 们在衡量一个函数运行时间,或者判断一个算法的时间效率,或者在程序中我们需要一个定时器,定时执行一个特定的操作,比如在多媒体中,比如在游戏中等,都 会用到时间函数.还比如我们通过记录函数或者算 ...

  2. windows时间函数

    介绍        我们在衡量一个函数运行时间,或者判断一个算法的时间效率,或者在程序中我们需要一个定时器,定时执 行一个特定的操作,比如在多媒体中,比如在游戏中等,都会用到时间函数.还比如我们通过记 ...

  3. MFC 相关类、函数

    timeSetEvent()函数 CRectTracker类的使用 SetLocalTime设置本地时间 AdjustTokenPrivileges启用权限

  4. VC++中的延时函数

    原文链接:http://www.educity.cn/develop/478947.html VC中提供了很多关于时间操作的函数,编写程序时我们可以跟据定时的不同精度要求选择不同的时间函数来完成定时和 ...

  5. VC中基于 Windows 的精确定时[转]

    在工业生产控制系统中,有许多需要定时完成的操作,如定时显示当前时间,定时刷新屏幕上的进度条,上位 机定时向下位机发送命令和传送数据等.特别是在对控制性能要求较高的实时控制系统和数据采集系统中,就更需要 ...

  6. [转载]win32 计时器使用

    在工业生产控制系统中,有许多需要定时完成的操作,如定时显示当前时间,定时刷新屏幕上的进度条,上位机定时向下位机发送命令和传送数据等.特别是在对控制性能要求较高的实时控制系统和数据采集系统中,就更需要精 ...

  7. windows获取时间的方法

    介绍       我们在衡量一个函数运行时间,或者判断一个算法的时间效率,或者在程序中我们需要一个定时器,定时执 行一个特定的操作,比如在多媒体中,比如在游戏中等,都会用到时间函数.还比如我们通过记录 ...

  8. VC++或QT下 高精度 多媒体定时器

    在VC编程中,用SetTimer可以定义一个定时器,到时间了,就响应OnTimer消息,但这种定时器精度太低了.如果需要精度更高一些的定时器(精 确到1ms),可以使用下面的高精度多媒体定时器进行代码 ...

  9. VC中基于 Windows 的精确定时

    在工业生产控制系统中,有许多需要定时完成的操作,如定时显示当前时间,定时刷新屏幕上的进度条,上位 机定时向下位机发送命令和传送数据等.特别是在对控制性能要求较高的实时控制系统和数据采集系统中,就更需要 ...

随机推荐

  1. tp3.2报错;syntax error, unexpected 'function' (T_FUNCTION), expecting identifier (T_STRING) or \\ (T_NS_SEPARATOR)

    出错原因:这个是php版本问题,laravel5.1的php版本要求是PHP >= 5.5.9,切换一下PHP版本就行.

  2. STM32CubeMx配置正交编码器遇到的问题

    配置时参考了这个哥们的方法: http://www.eemaker.com/stm32cubemx-encoder.html 然后我的配置是这样的 配置是没有问题. 调用时出现了问题. 由于配置完了, ...

  3. python中字典的遍历

    用ipython运行情况如下: #新建字典 In [1]: name_cards = {'name':'sunwukong','QQ':'123124','addr':'秦皇岛'} #生成key对象 ...

  4. Spring BindingResult验证框架Validation特殊用法

    使用注解@Valid(实体属性校验) Springboot实现 Spring实现 一.准备校验时使用的JAR validation-api-1.0.0.GA.jar:JDK的接口: hibernate ...

  5. SHIFT(文字列の指定位置数の移動)

    文字ごとの項目内容の移動 以下のような SHIFT 命令のバリアントを使用すると.項目内容を移動することができます.SHIFT を使用すると.文字ごとに項目内容が移動します. 文字列の指定位置数の移動 ...

  6. Create Fiori List App Report with ABAP CDS view – PART 2

    In the Part 1 blog, we have discussed below topics CDS annotations for Fiori List Report. How to cre ...

  7. Java基础——内部类

    一.什么是内部类 将一个类定义在另一个类里面或者一个方法里面,这样的类称为内部类 内部类所在的类在编译成功后,会出现这样两个class文件:OuterClass.class和OuterClass$In ...

  8. 使用USB Key(加密狗)实现身份认证

    首先你需要去买一个加密狗设备,加密狗是外形酷似U盘的一种硬件设备! 这里我使用的坚石诚信公司的ET99产品 公司项目需要实现一个功能,就是客户使用加密狗登录, 客户不想输入任何密码之类的东西,只需要插 ...

  9. [转][赞]Android开发者必知的开发资源

    英文原文:Bongzimo  翻译: ImportNew-黄小非 随着Android平台市场份额的持续猛增 ,越来越多的开发者开始投入Android应用程序的开发大潮.如果您是一位2013年刚刚入行的 ...

  10. Kindle 3(非常旧的版本) 隔一段时间自动重启问题

    买了本新书后,kindle 3 自己没事就在那边重启,几分钟一次 查到解决方案1: https://answers.yahoo.com/question/index?qid=2014040815565 ...