温馨提示×

温馨提示×

您好,登录后才能下订单哦!

密码登录×
登录注册×
其他方式登录
点击 登录注册 即表示同意《亿速云用户服务条款》

wxWidgets第十四课 wxTimer定时器

发布时间:2020-08-06 12:44:13 阅读:4144 作者:fengyuzaitu 栏目:系统运维
开发者测试专用服务器限时活动,0元免费领,库存有限,领完即止! 点击查看>>

说明

    OnIdle CPU空闲的情况下处理消息,如果需要定时功能,就需要使用定时器wxTimer 

问题

    比如定时器函数运行耗时10秒,定时的时间是10毫秒,是否是每隔10毫秒执行一次定时器函数,还是等待定时器函数运行结束,才开始重新计时

结果

    在定时器函数中执行::Sleep(10000);等待10秒,发现实际上需要等待定时器函数执行结束,才开始重新计时。所以在如下的场景需要特别小心:需要定时读取数据,而读取数据之后,执行一大堆耗时的操作,这个时候,就需要启动线程去处理,而不能在定时器函数中完成

例子

#include "wx/timer.h"

private:

wxTimer *m_timer;

//指定定时器的ID

#define TIMER_ID 1000

//将定时器ID和定时执行函数关联起来

EVT_TIMER(TIMER_ID, CFlightInstrumentPanel::OnTimer)

//创建定时器,指定定时器ID使用哪个定时器

m_timer = new wxTimer(this, TIMER_ID);

//启动定时器,参数是定时的时间间隔

m_timer->Start(1000);

编写定时器的执行内容

void CFlightInstrumentPanel::OnTimer( wxTimerEvent& event )

{

static int x = 0;

if (x<1000)

{

wxClientDC dc(this);

wxPen pen(*wxRED,1);

dc.SetPen(pen);

dc.DrawRectangle(x, 0, 200, 300);

dc.SetPen(wxNullPen);

x=x+100;

}

}

停止定时器

m_timer->Stop();

注意

定时器是一种资源,类似文件句柄,不可能无限的创建,定时结束之后,最后停止定时器,释放资源,并且如果在关闭窗口之前没有停止定时器,会出现

0xC0000005: 读取位置 0xFEEEFF06 时发生访问冲突错误,相关的内容查看其它的文章

定时器SetTimer的效率分析场景分析

    项目中使用wxWidgets框架,其中应用了该框架的定时器wxTimer,频繁进行了开启和关闭。在嵌入式操作系统中,性能是重中之重因此想尝试分析当前这种应用场景,是否会消耗CPU和内存的资源,跟踪wxWidgets的源码,发现定时器在windows系统下调用了使用了SetTimer和KillTimer函数进行定时器的启动和销毁。

疑惑

第一点:启动定时器是否是启动一条线程,然后Sleep等待时间的触发

第二点:频繁启动定时器,然后关闭,是否需消耗大量的资源

如果第一条成立的话,线程的创建以及切换都是非常

可观的开销

解惑

第一点:启动定时器SetTimer不是启动一个线程。该函数主要将新的

定时器结构加入内核的全局变量gptmrFirst这个链表,使用

KillTimer移除该定时器的结构体。系统会定时遍历该链表,

一旦定时时间就绪,就会向程序发送WM_TIMER消息,应用程序

接收到消息,开始处理逻辑

第二点:启动和关闭定时器也只是添加或者移除结构体,效率应该是

比较高的。创建线程的开销以及占用的堆栈都是可观的,尽管

可以设置线程堆栈的大小

前提

当前没有搜索到windows定时器的源码

参考:http://bbs.csdn.net/topics/360222963

基于以下的论断:

win32k中有一个全局变量gptmrFirst,里面存放了第一个定时器结构的指针,定时器结构以链表的形式储存

线程在消息循环中会调用GetMessageW->NtUserGetMessage->xxxInternalGetMessage->xxxRealInternalGetMessage

xxxRealInternalGetMessage后面会调用DoTimer,DoTimer就遍历整个定时器链表,并比较每个Timer的Win32Thread指针是不是于win32k的

全局变量gptiCurrent,gptiCurrent中保存的是当前线程的Win32Thread结构的指针(许多win32k函数开头都会有EnterCrit,这里面就设

置gptiCurrent为PsGetThreadWin32Thread的返回值)

如果是,表明这个Timer属于当前线程,所以就检查Timer是否已就绪(到时),如果就绪,则调用StoreQMessage放置一个WM_TIMER或

WM_SYSTIMER消息,后面xxxRealInternalGetMessage会将其取回

csrss.exe进程有一个叫raw input thread的内核线程,其内核对象地址的地址放在win32k全局变量gptiRit中

这个线程负责处理键盘输入,鼠标输入等,当然也有定时器,它会使用KeWaitForMultipleObjects等待一组内核对象,其中就有主定时

器,如果主定时器到时,就执行TimerProc

TimerProc函数会遍历gptmrFirst链表,减少每个定时器的剩余时间,并把到期的定时器设置为已就绪

亿速云「云服务器」,即开即用、新一代英特尔至强铂金CPU、三副本存储NVMe SSD云盘,价格低至29元/月。点击查看>>

向AI问一下细节

免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。

AI

开发者交流群×