timeSetEvent 使用方法及易出错的几种情况

好久没有写博客了,今天看见网上的资料好多都是一大抄,特写一篇自己调试的感受。

今天在调试项目的时候用到timeSetEvent,编译的时候没有出现错误,但运行的时候就会出现程序崩溃,在网上找了很多资料,参照MSDN,最终解决,特将遇到的问题和解决方法在此介绍,希望遇到同样问题的兄弟也能迅速定位问题。

查看MSDN,timeSetEvent的用法如下:

MMRESULT timeSetEvent( UINT uDelay, UINT uResolution,  LPTIMECALLBACK lpTimeProc, WORD dwUser, UINT fuEvent )
       其中: uDelay:以毫秒指定事件的周期。
              Uresolution:以毫秒指定延时的精度,数值越小定时器事件分辨率越高。缺省值为1ms。
              LpTimeProc:指向一个回调函数。
              DwUser:存放用户提供的回调数据。
              FuEvent:指定定时器事件类型:
              TIME_ONESHOT:uDelay毫秒后只产生一次事件
              TIME_PERIODIC :每隔uDelay毫秒周期性地产生事件。

参照函数的说明,函数很容易使用,但在使用的过程中应该注意以下几点,严格按照MSDN的介绍:

1,回调函数的使用

    在使用回调函数的时候,一定要注意回调函数的使用方法,MSDN上的说法如下,Pointer to a callback function that is called once upon expiration of a single event or periodically upon expiration of periodic events. If fuEventspecifies 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);注意红色部分在函数中一定不能丢掉,否则会引起程序的崩溃出现: 0x00000000 处有未经处理的异常: 0xC0000005: Access violation

2,timeKillEvent 关掉定时器的函数,一定要一一对应,每次timeSetEvent返回的定时器的ID是不一样的,也就是说调用一次timeSetEvent就会产生一次Id,就是说,调用几次timeSetEvent,就需要调用几次timeKillEvent ,而且必须是相对应的ID,否则可能出现程序崩溃!!!!!

------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------

具体的程序代码调试如下:

// timeset.cpp : 定义控制台应用程序的入口点。
//

#include "stdafx.h"
#include <windows.h>
#include <stdlib.h>
#include <stdio.h>

#pragma comment(lib,"Winmm.lib")
#define delaytime 1000

int  gtime_ID;

void  CALLBACK TimeEvent(UINT uTimerID, UINT uMsg, DWORD_PTR dwUser, DWORD_PTR dw1, DWORD_PTR dw2)
{
	printf("time ID is %d, started,dwUser is %d\n",gtime_ID, dwUser);
	return;
}

void StartEventTime(DWORD_PTR duser)
{
	gtime_ID = timeSetEvent(delaytime,10,(LPTIMECALLBACK)TimeEvent,duser,TIME_PERIODIC);
	if(gtime_ID == NULL)
	{
		printf("time ID is not created\n");
		return;
	}
	return;
}
int _tmain(int argc, _TCHAR* argv[])
{
	int i = 0;
	while (1)
	{
		StartEventTime(i);  
		Sleep(1100);
		i++;
		timeKillEvent(gtime_ID); 
		
		if (i == 10)
		{
			break;
		}
	}
	return 0;
}
程序中标红的部分一定要注意!!!就写到这里吧,希望对亲爱的兄弟姐妹有帮助,有什么问题,可以直接留言的,看到的一定会给予解答的,互相学习!!!!



  • 9
    点赞
  • 41
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
在 Delphi 11 64位中使用 timeSetEvent 函数可能会遇到一些问题,因为该函数是32位 Windows API 的一部分,而 Delphi 11 64位是基于 64位 Windows API 的。 为了在 Delphi 11 64位中使用 timeSetEvent 函数,你需要使用一个代替方案。一个可用的代替方案是使用 CreateTimerQueueTimer 函数。这个函数可以创建一个定时器,并在指定的时间间隔内重复调用一个回调函数。 以下是一个使用 CreateTimerQueueTimer 函数的示例代码: ``` var TimerQueue: THandle; TimerHandle: THandle; procedure TimerCallback(lpParameter: Pointer; TimerOrWaitFired: Boolean); begin // 处理定时器触发后的事件 end; // 创建定时器 procedure CreateTimer; begin TimerQueue := CreateTimerQueue; if TimerQueue <> 0 then begin // 创建定时器 if not CreateTimerQueueTimer(TimerHandle, TimerQueue, @TimerCallback, nil, 1000, 1000, 0) then RaiseLastOSError; end else RaiseLastOSError; end; // 销毁定时器 procedure DestroyTimer; begin DeleteTimerQueueTimer(TimerQueue, TimerHandle, INVALID_HANDLE_VALUE); DeleteTimerQueue(TimerQueue); end; ``` 在上面的代码中,我们首先使用 CreateTimerQueue 函数创建一个定时器队列。然后使用 CreateTimerQueueTimer 函数创建一个重复触发的定时器,并指定回调函数 TimerCallback。在回调函数中,我们可以处理定时器触发后的事件。最后,在程序退出时,我们调用 DeleteTimerQueueTimer 和 DeleteTimerQueue 函数销毁定时器。
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值