Delphi定时器控件TTimer“一睡不醒”问题研究
1,试验1—基础代码
1.1页面控件与代码
|
定时器 |
Timer1 |
Timer_work |
|
Interval |
1000 |
1500 |
|
Enabled |
True |
True |
|
Ontimer事件 |
if Timer1.Tag= then exit; //tag=1表示正在忙
Timer1.Tag := ;
try
memo1.Lines.Add(formatDatetime('NN:SS',now)+'Timer_work.tag=' + intToStr(Timer_work.tag));
finally
Timer1.Tag := ;
end;
|
if Timer_work.Tag= then exit; //tag=1表示正在忙
Timer_work.Tag := ;
try
memo1.Lines.Add(formatDatetime('NN:SS',now)+'Timer1.tag=' + intToStr(Timer1.tag));
finally
Timer_work.Tag := ;
end;
|
1.2运行结果与分析
39:15Timer_work.tag=0 39:16Timer1.tag=0 39:16Timer_work.tag=0 39:17Timer1.tag=0 39:17Timer_work.tag=0 39:19Timer_work.tag=0 39:19Timer1.tag=0 39:20Timer_work.tag=0 39:20Timer1.tag=0 39:21Timer_work.tag=0 39:22Timer_work.tag=0 39:22Timer1.tag=0 39:23Timer_work.tag=0 39:24Timer1.tag=0 39:24Timer_work.tag=0 39:25Timer_work.tag=0 39:25Timer1.tag=0 39:26Timer_work.tag=0 39:27Timer1.tag=0 39:27Timer_work.tag=0 39:28Timer_work.tag=0 39:28Timer1.tag=0 39:29Timer_work.tag=0 39:30Timer1.tag=0 39:30Timer_work.tag=0 39:31Timer_work.tag=0 39:31Timer1.tag=0 39:32Timer_work.tag=0 39:33Timer1.tag=0 39:33Timer_work.tag=0
以上为运行一段时间后memo1中的结果。其中:
Timer_work.tag=0出现的次数为17次
Timer1.tag=0出现的次数为12次
17/12约等于18/12=3:2=1.5:1
结果分析如下:
1,各个定时器均处于主线程中,可能是串行工作机制。即在Timer1的OnTimer事件中,Timer_work的OnTimer事件已经执行完毕,反之亦然,不会2个定时器事件同时处于执行状态。
2,2个定时器事件的执行次数比与定时器周期基本成反比。符合预期。
2,试验2—加入sleep和ProcessMessages后出现故障
2.1页面控件与代码
|
定时器 |
Timer1 |
Timer_work |
|
Interval |
1000 |
1500 |
|
Enabled |
True |
True |
|
Ontimer事件 |
if Timer1.Tag= then exit; //tag=1表示正在忙
Timer1.Tag := ;
try
application.ProcessMessages;
sleep(); memo1.Lines.Add(formatDatetime('NN:SS',now)+'Timer_work.tag=' + intToStr(Timer_work.tag));
finally
Timer1.Tag := ;
end;
|
if Timer_work.Tag= then exit; //tag=1表示正在忙
Timer_work.Tag := ;
try
application.ProcessMessages;
sleep(); memo1.Lines.Add(formatDatetime('NN:SS',now)+'Timer1.tag=' + intToStr(Timer1.tag));
finally
Timer_work.Tag := ;
end;
|
2.2运行结果与分析
02:33Timer_work.tag=0 02:34Timer1.tag=0 02:35Timer_work.tag=0 02:36Timer_work.tag=1 02:37Timer_work.tag=1 02:38Timer_work.tag=1 02:39Timer_work.tag=1 02:40Timer_work.tag=1 02:41Timer_work.tag=1 02:42Timer_work.tag=1 02:43Timer_work.tag=1 02:44Timer_work.tag=1 02:45Timer_work.tag=1 02:46Timer_work.tag=1 02:47Timer_work.tag=1 02:48Timer_work.tag=1 02:49Timer_work.tag=1 02:50Timer_work.tag=1 02:51Timer_work.tag=1 02:52Timer_work.tag=1 02:53Timer_work.tag=1 02:54Timer_work.tag=1 02:55Timer_work.tag=1 02:56Timer_work.tag=1 02:57Timer_work.tag=1 02:58Timer_work.tag=1 02:59Timer_work.tag=1 03:00Timer_work.tag=1 03:01Timer_work.tag=1
以上为运行一段时间后memo1中的结果。其中:
在Timer1的OnTimer事件中,出现了Timer_work.tag=1。这说明2个定时器事件同时处于执行状态。
以上事件出现后,就一直维持每一秒输出一次Timer_work.tag=1,说明程序一直在执行Timer1的OnTimer事件,而Timer_work的OnTimer事件被抑制。
结果分析如下:
1,各个定时器均处于主线程中,可能是串行工作机制。但并不意味着“在Timer1的OnTimer事件中,Timer_work的OnTimer事件已经执行完毕”,可能从一个定时器事件中跳出去执行另一个计时器事件,也就是说2个定时器事件可能同时处于事件响应过程中,但同一时间只有1个事件正在执行。
2,本身只准备睡眠1秒的Timer_work.,经过了25秒仍未恢复执行。而timer1调用周期为1秒加上sleep 1秒,应该是每2秒输出1次“Timer_work.tag=1”,但实际输出间隔是1秒,不符合预期。
3,试验3—故障分析
3.1页面控件与代码
|
定时器 |
Timer1 |
Timer_work |
|
Interval |
1000 |
1500 |
|
Enabled |
True |
True |
|
Ontimer事件 |
if Timer1.Tag= then exit; //tag=1表示正在忙
Timer1.Tag := ;
try
application.ProcessMessages;
memo1.Lines.Add(formatDatetime('NN:SS',now)+' in Timer1Timer before sleep' );
sleep(); memo1.Lines.Add(formatDatetime('NN:SS',now)+'Timer_work.tag=' + intToStr(Timer_work.tag));
finally
Timer1.Tag := ;
memo1.Lines.Add(formatDatetime('NN:SS',now)+'Timer1.tag=' + intToStr(Timer1.tag));
end;
|
if Timer_work.Tag= then exit; //tag=1表示正在忙
Timer_work.Tag := ;
try
application.ProcessMessages;
sleep();
|
停止按钮的响应事件:
procedure TForm1.Button1Click(Sender: TObject);
begin
memo1.Lines.Add(formatDatetime('NN:SS',now)+' 停止按钮已按下..' );
Timer_work.Enabled := False;
Timer1.Enabled := False;
memo1.Lines.Add(formatDatetime('NN:SS',now)+' 计时器已停止.' );
end;
3.2运行结果与分析
48:19 in Timer1Timer before sleep 48:20Timer_work.tag=1 48:20Timer1.tag=0 48:20 in Timer1Timer before sleep 48:21Timer_work.tag=1 48:21Timer1.tag=0 48:21 in Timer1Timer before sleep 48:22Timer_work.tag=1 48:22Timer1.tag=0 48:22 in Timer1Timer before sleep 48:23Timer_work.tag=1 48:23Timer1.tag=0 48:23 in Timer1Timer before sleep 48:24Timer_work.tag=1 48:24Timer1.tag=0 48:24 in Timer1Timer before sleep 48:25Timer_work.tag=1 48:25Timer1.tag=0 48:25 in Timer1Timer before sleep 48:26Timer_work.tag=1 48:26Timer1.tag=0 48:26 in Timer1Timer before sleep 48:27Timer_work.tag=1 48:27Timer1.tag=0 48:27 in Timer1Timer before sleep 48:28Timer_work.tag=1 48:28Timer1.tag=0 48:28 in Timer1Timer before sleep 48:29Timer_work.tag=1 48:29Timer1.tag=0 48:29 in Timer1Timer before sleep 48:30Timer_work.tag=1 48:30Timer1.tag=0 48:30 停止按钮已按下.. 48:30 计时器已停止. 48:31Timer1.tag=0 48:31Timer_work.tag=0
以上为运行一段时间后memo1中的结果。其中:
在Timer1的OnTimer事件中,sleep前后相差1秒,说明时间主要被sleep函数消耗,而不是被timer的周期消耗。
Timer_work.tag=0一直到停止该计时器时才出现。
结果分析如下:
1,timer控件的周期只是调用执行的时间间隔,实际执行时间需考虑多种因素。假设timer周期为T1,timer事件响应函数执行时间为T2,那么实际执行周期为max(T1,T2),而不是T1+T2。
2,多计时器同时启用时,可能出现无法想象的结果。应避免使用application.ProcessMessages和sleep的同时调用。
4,实验总结
实验证明,以下情景可能出现问题:
- 多个计时器函数中,至少有2个函数调用了application.ProcessMessages和 sleep中的1个或2个。
结论:
- 为避免计时器“一睡不醒”,在计时器函数中应该谨慎调用sleep函数。
Delphi定时器控件TTimer“一睡不醒”问题研究的更多相关文章
- <总结>delphi WebBrowser控件的使用中出现的bug
Delphi WebBrowser控件的使用中出现的bug: 1.WebBrowser.Visible=false:Visible属性不能使WebBrowser控件不可见,暂时用 WebBrowse ...
- 修改Delphi工具控件的默认字体
修改Delphi工具控件的默认字体: 注册表: Delphi 6: HKEY_CURRENT_USER\Software\Borland\Delphi\6.0Delphi 7: HKEY_ ...
- Delphi WebBrowser控件的使用(大全 good)
Delphi WebBrowser控件的使用 WebBrowser控件属性:1.Application 如果该对象有效,则返回掌管WebBrowser控件的应用程序实现的自动化对象(IDis ...
- Delphi TcxtreeList控件说明 转
Delphi TcxtreeList控件说明 树.cxTreeList 属性: Align:布局,靠左,靠右,居中等 AlignWithMargins:带边框的布局 Anchors:停靠 (akT ...
- delphi按钮控件的default属性
delphi按钮控件的default属性用于设置默认命令按钮,.设置为true时,按[Enter键]相当于用鼠标单击了该按钮 .窗口中如果有多个按钮的default是true的话,就根据tabinde ...
- Delphi fmx控件在手机滑动与单击的问题
Delphi fmx控件在手机滑动与单击的问题 (2016-03-08 10:52:00) 转载▼ 标签: it delphi 分类: Delphi10 众所周知,fmx制作的app,对于象TEdit ...
- Delphi maskedit控件的掩码含义及用法方法
Delphi maskedit控件的掩码含义及用法方法 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 2 ...
- Delphi StringGrid控件的用法
Delphi StringGrid控件 组件名称:StringGrid ●固定行及固定列: StringGrid.FixedCols:=固定行之数; StringGrid.Fixe ...
- Delphi IDHTTP控件:GET/POST 请求
Delphi IDHTTP控件:GET/POST 请求 最近一直在使用IDHTTP,下面是一些关于 GET.POST 请求基本使用方法的代码 一.GET 请求 1 procedure GetDem ...
随机推荐
- 织梦DEDE多选项筛选_联动筛选功能的实现_二次开发
织梦默认的列表页没有筛选功能,但有时候我们做产品列表页的时候,产品的字段比较多,很多人都需要用到筛选功能,这样可以让用户更方便的找到自己所需要的东西,实现这个联动筛选功能需要对织梦进行二次开发,下面就 ...
- 在织梦dedecms中实现“文章标题-栏目名称-网站名”导航
本文介绍了在dedecms中,实现文章标题-栏目名称-网站名 导航的方法,有需要的朋友参考下. 在dedecms中实现“文章标题-栏目名称-网站名”导航的方法. 第一种: 在/include/in ...
- 星星dom
<!DOCTYPE html><html><head><meta charset="UTF-8"><title>星星&l ...
- 官网下载java相关资源
官网下载java相关资源 官网地址:http://www.oracle.com 一.下载JDK 1.首先进入Downloads >> Java For Developers,如图 2.点击 ...
- MP4V2库与MP4AV库编译
最近在开发一个将RTP流存储为MP4文件的功能.其中针对MP4文件读写,用到了两个开源的库.其中MP4V2用于数据的读写,MP4AV用于对其中的数据帧进行分析. MP4V2和MP4AV都是开源项目MP ...
- nc之二:nc命令详解
NetCat,在网络工具中有“瑞士军刀”美誉,其有Windows和Linux的版本.因为它短小精悍(1.84版本也不过25k,旧版本或缩减版甚至更小).功能实用,被设计为一个简单.可靠的网络工具,可通 ...
- 用 SDL2 进行事件驱动编程
其实没必要说得太复杂...就是读取用户输入啦. 沿用上一篇的代码,加入事件轮询. 环境:SDL2 + VC++2015 下面的代码将打开background.png和event.png,将backgr ...
- 【旧文章搬运】ZwQuerySystemInformation枚举进线程信息
原文发表于百度空间,2008-10-15========================================================================== 很古老的东 ...
- Ruby module ---模块,组件
module 的主要目的是把不同的方法和常量分别放进不同的命名空间. module 的命名方式跟类一样首字母大写,多个单词不用下划线. 如:CircleArea module 语法 module Mo ...
- mysql5.7根据.frm和.ibd文件恢复表结构和数据
一.恢复表结构 1.环境:Windows .mysql5.7:首先创建一个数据库,可以通过navicat来创建: 2.使用当前创建的数据库:use ww; 3.随意创建一张表,但是这张表的名字 ...