温馨提示×

温馨提示×

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

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

wxWidgets第十四课 wxTimer定时器

发布时间:2020-08-06 12:44:13 来源:网络 阅读:4140 作者:fengyuzaitu 栏目:系统运维

说明

    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链表,减少每个定时器的剩余时间,并把到期的定时器设置为已就绪


向AI问一下细节

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

AI