delphi 线程池基础 TSimplePool
1. TSimpleThread
2. TSimpleList
3. 以1,2构成 TSimplePool
用法
先定义: TDoSomeThingThread=class(TSimpleThread) ;
并给 TDoSomeThingThread reintroduce Create 不带参数的构造函数。
再定义 TDoSomeThingPool=class(TSimpleTool<TDoSomeThing>);
最后,只需在 TDoSomeThingPool 写线程调度的代码就行了,可以省不少事。(这部分有待进一步完善)
unit uSimpleThread;
interface
uses
System.Classes, System.SysUtils, System.SyncObjs; type // 显示信息,调用方法 DoOnStatusMsg(AMsg);
TOnStatusMsg = procedure(AMsg: string) of object; // 显示调试信息,一般用于显示出错信息,用法 DoOnDebugMsg(AMsg);
TOnDebugMsg = TOnStatusMsg; TSimpleThread = class(TThread)
public type // "执行过程"的类别定义 TGeneralProc = procedure; // 普通的,即 procedure DoSomeThing;
TObjectProc = procedure of object; // 类的,即 TXxxx.DoSomeThign; 用得多
TAnonymousProc = reference to procedure; // 匿名的
private type
TProcKind = (pkGeneral, pkObject, pkAnonymous); // "执行过程"的类别
private FGeneralProc: TGeneralProc;
FObjProc: TObjectProc;
FAnoProc: TAnonymousProc; FProcKind: TProcKind; FEvent: TEvent; // 用于阻塞,它是一个信号量
FActiveX: boolean; // 是否在线程中支持 Com ,如果你要在线程中访问 IE 的话,就设定为 True FOnStatusMsg: TOnStatusMsg;
FOnDebugMsg: TOnDebugMsg; FTagID: integer; // 给线程一个代号,在线程池的时候用来作区别
FParam: integer; // 给线程一个参数,方便识别 procedure SelfStart; // 触发线程运行 procedure DoExecute; // 这个函数里面运行的代码是“线程空间”
procedure DoOnException(e: exception); // 异常信息显示 调用 DoOnDebugMsg(AMsg); procedure SetTagID(const Value: integer);
procedure SetParam(const Value: integer); procedure SetOnStatusMsg(const Value: TOnStatusMsg);
procedure SetOnDebugMsg(const Value: TOnDebugMsg); protected FWaitStop: boolean; // 结束标志,可以在继承类中使用它,以确定线程是否停止运行 procedure DoOnStatusMsg(AMsg: string); // 显示普通信息
procedure DoOnDebugMsg(AMsg: string); // 显示调式信息 procedure Execute; override; // 重载 TThread.Execute procedure OnThreadProcErr(e: exception); virtual; // 异常发生事件 procedure WaitThreadStop; // 等待线程结束 procedure BeforeExecute; virtual; // 看名字,不解释
Procedure AfterExecute; virtual; // 看名字,不解释 procedure SleepExceptStopped(ATimeOut: Cardinal); // 这个高大上了,要解释一下。
{ 有时线程没有任务时,就会休息一会儿,但是,休息的时候,可能会接收到退出线程的指令
此函数就是在休息的时候也检查一下停止指令
} public // 改变一下 Create 的参数,AllowedActiveX:是否允许线程代码访问 Com
constructor Create(AllowedActiveX: boolean = false); reintroduce; destructor Destroy; override; procedure ExeProcInThread(AProc: TGeneralProc); overload; // 这三个,对外的接口。
procedure ExeProcInThread(AProc: TObjectProc); overload;
procedure ExeProcInThread(AProc: TAnonymousProc); overload; procedure StartThread; virtual;
{ 启动线程,一般只调用一次。
以后就由线程的响应事件来执行了
} procedure StopThread; virtual; // 停止线程 property OnStatusMsg: TOnStatusMsg read FOnStatusMsg write SetOnStatusMsg;
property OnDebugMsg: TOnDebugMsg read FOnDebugMsg write SetOnDebugMsg;
property WaitStop: boolean read FWaitStop;
property TagID: integer read FTagID write SetTagID;
property Param: integer read FParam write SetParam; end; implementation uses
ActiveX; procedure TSimpleThread.AfterExecute;
begin
end; procedure TSimpleThread.BeforeExecute;
begin
end; constructor TSimpleThread.Create(AllowedActiveX: boolean);
var
BGUID: TGUID;
begin
inherited Create(false);
FActiveX := AllowedActiveX;
FreeOnTerminate := false; // 我们要手动Free线程
CreateGUID(BGUID);
FEvent := TEvent.Create(nil, true, false, GUIDToString(BGUID));
end; destructor TSimpleThread.Destroy;
begin
StopThread; // 先停止
WaitThreadStop; // 再等待线程停止
{
在继承类的 Destroy 中,也要写上这两句. 如:
暂时未找到更好的办法,这点代码省不了
destructor TXXThread.Destroy;
begin
StopThread;
WaitThreadStop;
xxx.Free;
Inherited;
end;
}
FEvent.Free;
inherited;
end; procedure TSimpleThread.DoExecute; // 此函数内执行的代码,就是在多线程空间里运行
begin
BeforeExecute;
repeat FEvent.WaitFor;
FEvent.ResetEvent; // 下次waitfor 一直等
{ 这里尝试了很多些,总 SelfStart 觉得有冲突,经过多次修改并使用证明,
没有必要在这里加锁,因为只调用 startThread 一次,剩下的交给线程影应事件
} if not Terminated then // 如果线程需要退出
begin try case FProcKind of
pkGeneral: FGeneralProc;
pkObject: FObjProc;
pkAnonymous: FAnoProc;
end; except on e: exception do
begin
DoOnException(e);
end; end; end; until Terminated;
AfterExecute;
//代码运行到这里,就表示这个线程不存在了。再也回不去了,必须释放资源了。
end; procedure TSimpleThread.DoOnDebugMsg(AMsg: string);
begin
if Assigned(FOnDebugMsg) then
FOnDebugMsg(AMsg);
end; procedure TSimpleThread.DoOnException(e: exception);
var
sErrMsg: string;
begin
sErrMsg := 'ClassName:' + ClassName + ##;
sErrMsg := sErrMsg + 'TagID:' + IntToStr(FTagID) + ##;
sErrMsg := sErrMsg + 'Param:' + IntToStr(Param) + ##;
sErrMsg := sErrMsg + 'ErrMsg:' + e.Message + ##;
DoOnDebugMsg(sErrMsg);
OnThreadProcErr(e);
end; procedure TSimpleThread.DoOnStatusMsg(AMsg: string);
begin
if Assigned(FOnStatusMsg) then
FOnStatusMsg(AMsg);
end; procedure TSimpleThread.Execute;
begin
//是否支持 Com
if FActiveX then
begin
CoInitialize(nil);
try
DoExecute;
finally
CoUninitialize;
end;
end
else
DoExecute;
end; procedure TSimpleThread.ExeProcInThread(AProc: TGeneralProc);
begin
FGeneralProc := AProc;
FProcKind := pkGeneral;
SelfStart;
end; procedure TSimpleThread.ExeProcInThread(AProc: TObjectProc);
begin
FObjProc := AProc;
FProcKind := pkObject;
SelfStart;
end; procedure TSimpleThread.ExeProcInThread(AProc: TAnonymousProc);
begin
FAnoProc := AProc;
FProcKind := pkAnonymous;
SelfStart;
end; procedure TSimpleThread.OnThreadProcErr(e: exception);
begin;
end; procedure TSimpleThread.SelfStart;
begin
//经常多次尝试,最终写成这样,运行没有问题
if FEvent.WaitFor() <> wrSignaled then
FEvent.SetEvent; // 让waitfor 不再等
end; procedure TSimpleThread.StopThread;
begin
//继承类的代码中,需要检查 FWaitStop ,来控制线程结束
FWaitStop := true;
end; procedure TSimpleThread.SetOnDebugMsg(const Value: TOnDebugMsg);
begin
FOnDebugMsg := Value;
end; procedure TSimpleThread.SetOnStatusMsg(const Value: TOnStatusMsg);
begin
FOnStatusMsg := Value;
end; procedure TSimpleThread.SetParam(const Value: integer);
begin
FParam := Value;
end; procedure TSimpleThread.SetTagID(const Value: integer);
begin
FTagID := Value;
end; procedure TSimpleThread.SleepExceptStopped(ATimeOut: Cardinal);
var
BOldTime: Cardinal;
begin
// sleep 时检测退出指令,以确保线程顺序退出
// 多个线程同时工作,要保证正确退出,确实不容易
BOldTime := GetTickCount;
while not WaitStop do
begin
sleep();
if (GetTickCount - BOldTime) > ATimeOut then
break;
end;
end; procedure TSimpleThread.StartThread;
begin
FWaitStop := false;
end; procedure TSimpleThread.WaitThreadStop;
begin
//等待线程结束
StopThread;
Terminate;
SelfStart;
WaitFor;
end; end.
uSimpleThread.pas
unit uSimpleList; interface uses
Generics.Collections; type TSimpleList<T> = class(TList<T>)
private
FCurIndexPos: integer;
function DoPopByIndex(Index: integer): T;
procedure FreeAllItems;
procedure SetCurIndexPos(const Value: integer);
protected
FNeedFreeItem: boolean;
procedure FreeItem(Item: T); virtual; //子类可以重截这个以确定该如何释放
public constructor Create;
destructor Destroy; override; procedure Lock; //新版的Lock功能值得学习
procedure Unlock; // function PopFirst: T; //不解释,下同
function PopLast: T;
function PopByIndex(Index: integer): T; procedure ClearAndFreeAllItems; //清空并释放所有的Item
property CurIndexPos: integer read FCurIndexPos write SetCurIndexPos; end; //加 Constructor 限制是要求 T 要有一个没带参数的Create函数,也就是构造器
TClassSimpleList<T: Class, Constructor> = class(TSimpleList<T>)
protected
procedure FreeItem(Item: T); override;
function AddNewOne: T;// T有了Create 才能写这个
end; implementation procedure TSimpleList<T>.ClearAndFreeAllItems;
begin
FreeAllItems;
clear;
end; constructor TSimpleList<T>.Create;
begin
inherited;
FNeedFreeItem := true;
FCurIndexPos := -;
end; destructor TSimpleList<T>.Destroy;
begin
FreeAllItems;
inherited;
end; function TSimpleList<T>.DoPopByIndex(Index: integer): T;
begin
if (index >= ) and (index <= count - ) then
begin
result := items[index];
Delete(index);
Exit;
end;
result := T(nil);
end; procedure TSimpleList<T>.FreeAllItems;
var
Item: T;
begin
if FNeedFreeItem then
begin
FCurIndexPos := -;
for Item in self do
FreeItem(Item);
end;
end; procedure TSimpleList<T>.FreeItem(Item: T);
begin
// 假设 T 是 PMyRec =^TMyRec TMyRec=record;
// 这个写法对吗?
// if GetTypeKind(T) = tkPointer then
// begin
// Dispose(Pointer(Pointer(@Item)^));
// end;
// 此写法未认真测试所以不使用。
// 如果 Item 是指针,我在继承类中的 FreeItem 中写 Dispose(Item);
end; procedure TSimpleList<T>.Lock;
begin
system.TMonitor.Enter(self);
end; procedure TSimpleList<T>.Unlock;
begin
system.TMonitor.Exit(self);
end; function TSimpleList<T>.PopByIndex(Index: integer): T;
begin
result := DoPopByIndex(index);
end; function TSimpleList<T>.PopFirst: T;
begin
result := DoPopByIndex();
end; function TSimpleList<T>.PopLast: T;
begin
result := DoPopByIndex(count - );
end; procedure TSimpleList<T>.SetCurIndexPos(const Value: integer);
begin
FCurIndexPos := Value;
end; { TThreadClassList<T> } function TClassSimpleList<T>.AddNewOne: T;
begin
result := T.Create();
Add(result);
end; procedure TClassSimpleList<T>.FreeItem(Item: T);
begin
Item.Free;
end; end.
uSimpleList.pas
unit uSimplePool; interface uses
uSimpleThread, uSimpleList, uSyncObjs, System.Generics.Collections; Type TSimplePool<T: TSimpleThread, Constructor> = class
private Type
TWorkThreadList = Class(TClassSimpleList<T>);
private FOnStatusMsg: TOnStatusMsg;
FOnDebugMsg: TOnDebugMsg;
FMaxThreadCount: integer; procedure SetOnDebugMsg(const Value: TOnDebugMsg);
procedure SetOnStatusMsg(const Value: TOnStatusMsg);
procedure SetMaxThreadCount(const Value: integer);
procedure InitThreadList(AThreadCount: integer); protected FStopThreadCount: integer;
FWorkThreadList: TWorkThreadList;
FEvent: TSuperEvent; //提供给继承类阻塞用 procedure DoOnStatusMsg(AMsg: string);
procedure DoOnDebugMsg(AMsg: string);
procedure OnEachNewWorkThread(AWorkThread: T); virtual; public property OnStatusMsg: TOnStatusMsg read FOnStatusMsg write SetOnStatusMsg;
property OnDebugMsg: TOnDebugMsg read FOnDebugMsg write SetOnDebugMsg; constructor Create;
destructor Destroy; override; procedure StartWork; virtual;
procedure StopWork; virtual; property MaxThreadCount: integer read FMaxThreadCount write SetMaxThreadCount default ; end; const
cnDefaultWorkThreadCount = ;
cnLimitedWorkTreadCount = ; implementation { TSimplePool } procedure TSimplePool<T>.DoOnDebugMsg(AMsg: string);
begin
if Assigned(FOnDebugMsg) then
FOnDebugMsg(AMsg);
end; procedure TSimplePool<T>.DoOnStatusMsg(AMsg: string);
begin
if Assigned(FOnStatusMsg) then
FOnStatusMsg(AMsg);
end; procedure TSimplePool<T>.InitThreadList(AThreadCount: integer);
var
i, nTagID: integer;
B: T;
begin
nTagID := FWorkThreadList.Count;
for i := to AThreadCount do
begin
B := FWorkThreadList.AddNewOne;
B.TagID := nTagID;
B.OnStatusMsg := self.DoOnStatusMsg;
B.OnDebugMsg := self.DoOnDebugMsg;
OnEachNewWorkThread(B);
inc(nTagID);
end;
end; procedure TSimplePool<T>.OnEachNewWorkThread(AWorkThread: T);
begin
end; procedure TSimplePool<T>.SetMaxThreadCount(const Value: integer);
var
ndiff: integer;
begin
FMaxThreadCount := Value;
if FMaxThreadCount > cnLimitedWorkTreadCount then
FMaxThreadCount := cnLimitedWorkTreadCount;
if FMaxThreadCount <= then
FMaxThreadCount := ;
ndiff := FMaxThreadCount - FWorkThreadList.Count;
InitThreadList(ndiff);
end; procedure TSimplePool<T>.SetOnDebugMsg(const Value: TOnDebugMsg);
begin
FOnDebugMsg := Value;
end; procedure TSimplePool<T>.SetOnStatusMsg(const Value: TOnStatusMsg);
begin
FOnStatusMsg := Value;
end; procedure TSimplePool<T>.StartWork;
var
i: integer;
begin
for i := to MaxThreadCount do
begin
FWorkThreadList[i].StartThread;
end;
end; procedure TSimplePool<T>.StopWork;
var
B: T;
begin
for B in FWorkThreadList do
begin
B.StopThread;
end;
end; constructor TSimplePool<T>.Create;
begin
inherited Create;
FMaxThreadCount := ;
FEvent := TSuperEvent.Create;
FWorkThreadList := TWorkThreadList.Create;
InitThreadList(cnDefaultWorkThreadCount);
end; destructor TSimplePool<T>.Destroy;
begin
FWorkThreadList.Free;
FEvent.Free;
inherited Destroy;
end; end.
uSimplePool.pas
unit uSyncObjs; interface uses
SyncObjs; Type TSuperEvent = class(TEvent)
public
constructor Create; reintroduce;
end; implementation { TSuperEvent }
uses
SysUtils; constructor TSuperEvent.Create;
var
BGUID: TGUID;
begin
CreateGUID(BGUID);
inherited Create(nil, true, false, GUIDToString(BGUID));
end; end.
uSyncObjs.pas
delphi 线程池基础 TSimplePool的更多相关文章
- C#线程池基础
池(Pool)是一个很常见的提高性能的方式.比如线程池连接池等,之所以有这些池是因 为线程和数据库连接的创建和关闭是一种比较昂贵的行为.对于这种昂贵的资源我们往往会考虑在一个池容器中放置一些资源,在用 ...
- Java 多线程(五)—— 线程池基础 之 FutureTask源码解析
FutureTask是一个支持取消行为的异步任务执行器.该类实现了Future接口的方法. 如: 取消任务执行 查询任务是否执行完成 获取任务执行结果(”get“任务必须得执行完成才能获取结果,否则会 ...
- c# 多线程线程池基础
线程池的作用 在上一篇中我们了解了创建和销毁线程是一个昂贵的操作,要耗费大量的时间,太多的线程会浪费内存资源,当线程数量操作计算机CPU的数量后操作系统必须调度可运行的线程并执行上下文切 ...
- Java线程池基础
目录: 一.线程池概述 二.线程池参数 三.线程池的执行过程 四.线程池的主要实现 五.线程池的使用 六.线程池的正确关闭方式 七.线程池参数调优 一.线程池概述 1.线程池类 目前线程池类一般有两个 ...
- JUC之线程池基础
线程池 定义和方法 线程池的工作时控制运行的线程数量,处理过程中将任务放入队列,然后在线程创建后启动这些任务,如果线程数量超过了最大数量,超出数量的线程排队等候,等待其他线程执行完成,再从队列中取出任 ...
- JUC之线程池基础与简单源码分析
线程池 定义和方法 线程池的工作时控制运行的线程数量,处理过程中将任务放入队列,然后在线程创建后启动这些任务,如果线程数量超过了最大数量,超出数量的线程排队等候,等待其他线程执行完成,再从队列中取出任 ...
- Delphi线程池
unit uThreadPool; { aPool.AddRequest(TMyRequest.Create(RequestParam1, RequestParam2, ...)); } inte ...
- Java 基础 多线程和线程池基础
一,多线程 1.1 多线程介绍 进程:进程指正在运行的程序.确切的来说,当一个程序进入内存运行,即变成一个进程,进程是处于运行过程中的程序,并且具有一定独立功能. 线程:线程是进程中的一个执行单元,负 ...
- Java线程和多线程(十二)——线程池基础
Java 线程池管理多个工作线程,其中包含了一个队列,包含着所有等待被执行的任务.开发者可以通过使用ThreadPoolExecutor来在Java中创建线程池. 线程池是Java中多线程的一个重要概 ...
随机推荐
- 库函数strlen源码重现及注意问题
首先直接上源码: size_t strlen (const char * str) { const char *eos = str; while(*eos++); return(eos - str - ...
- php学习网址
后面会陆续维护此页. 1. php编程 此博客是网站www.beilei123.cn镜像,转载请注明出处.
- gulp配置版本号 教程之gulp-rev-append
简介: 使用gulp-rev-append给页面的引用添加版本号,清除页面引用缓存. 1.安装nodejs/全局安装gulp/项目安装gulp/创建package.json和gulpfile.js文件 ...
- django是怎么处理请求的
本文摘自 http://djangobook.py3k.cn/2.0/chapter03/ 我们在Django建立helloworld自定义页面中新建了站点,并能接受URL请求展示我们的页面,那Dja ...
- 初学swift笔记变量的定义(一)
swift变量的定义 1 import Foundation /* 变量的定义 变量的类型是可以不用写的 var a=10 常量的定义 let修饰 */ print(a) let b= print(b ...
- 安装程序时出现错误代码0x80070422
通过win10应用商店,下载应用,安装时出现错误代码0x80070422. 需要打开services.msc,将windows update服务打开.
- CxImage的使用
1.首先从此处下载源代码 http://www.codeproject.com/KB/graphics/cximage.aspx 2.然后将里面的工程全部编译一下,我觉得应该是生成对应的库. 3.然后 ...
- 使用kd-tree加速k-means
0.目录 前置知识 思路介绍 详述 1 确定h的中心点 2 算法步骤 java实现 1.前置知识 本文内容基于<Accelerating exact k-means algorithms wit ...
- 实现在ios文件读写
文件都是用来读写数据的,可是哪里都会有潜规则,ios里面读写数据的潜规则你知不知道,你知道不知道!!! 你有没有觉得NSUserDefaults和NSBundle,plist这些玩意阴魂不散,有时候搞 ...
- [Linux] 解压缩 tar 命令详解
在Linux环境软件安装过程中通常需要用到解压命令,故在此总结下,以方便以后使用,若有不对之处,欢迎指正. 1. 文件压缩 通过压缩算法将文件的体积缩小,同时会将多个文件合并成至一起方便 ...