多线程编程(17) - 多线程同步之 WaitableTimer (等待定时器对象)[续三]
2021/4/30 14:26:49
本文主要是介绍多线程编程(17) - 多线程同步之 WaitableTimer (等待定时器对象)[续三],对大家解决编程问题具有一定的参考价值,需要的程序猿们随着小编来一起学习吧!
多线程编程(17) - 多线程同步之 WaitableTimer (等待定时器对象)[续三]
根据 WaitableTimer 的主要功用, 现在再把它放在 "线程同步" 的话题中讨论有点不合适了, 就要结束它.
//重新看看那个 APC 回调函数的格式: procedure TimerAPCProc( lpArgToCompletionRoutine: Pointer; dwTimerLowValue, dwTimerHighValue: DWORD ); stdcall;
TimerAPCProc 的后两个参数其实是在传递一个值, 使用时要把它们合并为一个 TFileTime 类型的时间.
这个时间是 APC 函数被调用的时间, 稍稍修改上面一个例子看一下:
代码文件:
unit Unit1; interface uses Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms, Dialogs, StdCtrls; type TForm1 = class(TForm) Button1: TButton; Button2: TButton; procedure Button1Click(Sender: TObject); procedure Button2Click(Sender: TObject); procedure FormDestroy(Sender: TObject); end; var Form1: TForm1; implementation {$R *.dfm} var hTimer: THandle; {APC 函数} procedure TimerAPCProc(lpArgToCompletionRoutine: Pointer; dwTimerLowValue: DWORD; dwTimerHighValue: DWORD); stdcall; var UTCFileTime,LocalFileTime: TFileTime; SystemTime: TSystemTime; DateTime: TDateTime; begin {把 dwTimerLowValue 与 dwTimerHighValue 和并为一个 TFileTime 格式的时间} UTCFileTime.dwLowDateTime := dwTimerLowValue; UTCFileTime.dwHighDateTime := dwTimerHighValue; FileTimeToLocalFileTime(UTCFileTime, LocalFileTime); {从世界标准计时到本地时间} FileTimeToSystemTime(LocalFileTime, SystemTime); {转到系统格式时间} DateTime := SystemTimeToDateTime(SystemTime); {再转到 TDateTime} Form1.Text := DateTimeToStr(DateTime); SleepEx(INFINITE, True); end; {线程入口函数} function MyThreadFun(p: Pointer): Integer; stdcall; var DueTime: Int64; begin DueTime := 0; if SetWaitableTimer(hTimer, DueTime, 1000, @TimerAPCProc, nil, False) then begin SleepEx(INFINITE, True); end; Result := 0; end; procedure TForm1.Button1Click(Sender: TObject); var ID: DWORD; begin if hTimer = 0 then hTimer := CreateWaitableTimer(nil, True, nil); CreateThread(nil, 0, @MyThreadFun, nil, 0, ID); end; procedure TForm1.Button2Click(Sender: TObject); begin CancelWaitableTimer(hTimer); end; procedure TForm1.FormDestroy(Sender: TObject); begin CloseHandle(hTimer); end; end.
窗体文件:
object Form1: TForm1 Left = 0 Top = 0 Caption = 'Form1' ClientHeight = 86 ClientWidth = 256 Color = clBtnFace Font.Charset = DEFAULT_CHARSET Font.Color = clWindowText Font.Height = -11 Font.Name = 'Tahoma' Font.Style = [] OldCreateOrder = False PixelsPerInch = 96 TextHeight = 13 object Button1: TButton Left = 23 Top = 32 Width = 97 Height = 25 Caption = #21551#21160#23450#26102#22120 TabOrder = 0 OnClick = Button1Click end object Button2: TButton Left = 134 Top = 32 Width = 97 Height = 25 Caption = #21462#28040#23450#26102#22120 TabOrder = 1 OnClick = Button2Click end end
SetWaitableTimer 中回调函数后面的指针参数, 将被传递给 APC 函数的第一个参数;
作为指针它可以携带任何数据, 这里让它携带了一个坐标点(鼠标点击窗体的位置), 下例效果图:
代码文件:
unit Unit1; interface uses Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms, Dialogs, StdCtrls; type TForm1 = class(TForm) procedure FormDestroy(Sender: TObject); procedure FormMouseDown(Sender: TObject; Button: TMouseButton; Shift: TShiftState; X, Y: Integer); end; var Form1: TForm1; implementation {$R *.dfm} var hTimer: THandle; pt: TPoint; {APC 函数} procedure TimerAPCProc(lpArgToCompletionRoutine: Pointer; dwTimerLowValue: DWORD; dwTimerHighValue: DWORD); stdcall; var UTCFileTime,LocalFileTime: TFileTime; SystemTime: TSystemTime; DateTime: TDateTime; pt2: TPoint; begin UTCFileTime.dwLowDateTime := dwTimerLowValue; UTCFileTime.dwHighDateTime := dwTimerHighValue; FileTimeToLocalFileTime(UTCFileTime, LocalFileTime); FileTimeToSystemTime(LocalFileTime, SystemTime); DateTime := SystemTimeToDateTime(SystemTime); pt2 := PPoint(lpArgToCompletionRoutine)^; {接受指针参数} Form1.Canvas.Lock; Form1.Canvas.TextOut(pt2.X, pt2.Y, DateTimeToStr(DateTime)); Form1.Canvas.Unlock; SleepEx(INFINITE, True); end; {线程入口函数} function MyThreadFun(p: Pointer): Integer; stdcall; var DueTime: Int64; begin DueTime := 0; {参数 @pt 在这里是鼠标点击窗体时的坐标结构的指针, 它将传递给 APC 函数的第一个参数} if SetWaitableTimer(hTimer, DueTime, 1000, @TimerAPCProc, @pt, False) then begin SleepEx(INFINITE, True); end; Result := 0; end; {建立 WaitableTimer 对象和线程} procedure TForm1.FormMouseDown(Sender: TObject; Button: TMouseButton; Shift: TShiftState; X, Y: Integer); var ID: DWORD; begin pt := Point(X,Y); {在这里个全局的坐标点赋值} if hTimer = 0 then hTimer := CreateWaitableTimer(nil, True, nil); CreateThread(nil, 0, @MyThreadFun, nil, 0, ID); end; procedure TForm1.FormDestroy(Sender: TObject); begin CloseHandle(hTimer); end; end.
窗体文件:
object Form1: TForm1 Left = 0 Top = 0 Caption = 'Form1' ClientHeight = 135 ClientWidth = 195 Color = clBtnFace Font.Charset = DEFAULT_CHARSET Font.Color = clWindowText Font.Height = -11 Font.Name = 'Tahoma' Font.Style = [] OldCreateOrder = False onm ouseDown = FormMouseDown PixelsPerInch = 96 TextHeight = 13 end
posted on 2009-02-19 12:19 万一 阅读(5945) 评论(5) 编辑 收藏
这篇关于多线程编程(17) - 多线程同步之 WaitableTimer (等待定时器对象)[续三]的文章就介绍到这儿,希望我们推荐的文章对大家有所帮助,也希望大家多多支持为之网!
- 2024-11-20实战:30 行代码做一个网页端的 AI 聊天助手
- 2024-11-185分钟搞懂大模型的重复惩罚后处理
- 2024-11-18基于Ollama和pgai的个人知识助手项目:用Postgres和向量扩展打造智能数据库
- 2024-11-15我用同一个提示测试了4款AI工具,看看谁设计的界面更棒
- 2024-11-15深度学习面试的时候,如何回答1x1卷积的作用
- 2024-11-15检索增强生成即服务:开发者的得力新帮手
- 2024-11-15技术与传统:人工智能时代的最后一袭纱丽
- 2024-11-15未结构化数据不仅仅是给嵌入用的:利用隐藏结构提升检索性能
- 2024-11-15Emotion项目实战:新手入门教程
- 2024-11-157 个开源库助你构建增强检索生成(RAG)、代理和 AI 搜索