(本章节中例子都是用 VS2010 编译调试的)
线程池编写必须在 Windows Vista 操作系统(以及以上版本的操作系统)下,且 C++ 编译器版本至少是 VS2008
线程池的功能
注意
当一个进程初始化的时候,它并没有任何与线程池的开销.但是,一旦调用了新的线程池函数,系统就会为进程相应的内核资源,其中的一些资源在进程终止之前都将一直存在.正如我们可以看到,使用线程池的开销取决于用法:系统会以进程的名义来分配线程,其他内核以及内部数据结构.因此我们不应该盲目地使用这些线程池函数,而是必须谨慎地考虑,这些函数能做什么,以及它们不能做什么.
在线程池编程中,我们从来不需要自己调用 CreateThread.系统会自动为我们的进程创建线程,并在规定的条件下让线程池中的线程调用我们的回调函数.此外,这个线程在处理完成一个客户请求后,它不会立刻被销毁,而是回到线程池,准备好处理队列中的任何其他工作项,线程池会不断地重复使用其中的线程,而不会频繁地创建销毁线程,对应用程序来说,这样可以显著地提升性能,因为创建和销毁线程会消耗大量的时间.当然,如果线程池检测到创建的另一个线程将能更好地为应用程序服务,那么它会这样做.如果线程池检测到它的线程数量已经供过于求,那么它会销毁其中一些线程.除非我们非常清楚自己在做什么,否则的话最好还是相信线程内部的算法,让它自动地对应用程序的工作量进行微调.
默认线程池,在进程存在期间它不会被销毁.生命周期与进程相同.在进程终止的时候,Windows 会将其销毁并负责所有的清理工作.
对线程池的制定
可以用 CreateThreadpool 来创建一个新的线程池,该函数返回一个 PTP_POOL 值,表示新创建的线程池.接着我们可以调用后面两个函数来设置线程池中线程的最大数量和最小数量 SetThreadpoolThreadMinimum / SetThreadpoolThreadMaximum 线程池始终保持池中的线程数量至少是指定的最小数量,并允许线程数量增长到指定的最大数量,顺便一提,默认线程池的最小数量为1,最大数量为500.然后在引用程序不在需要自定义线程池时,应调用 CloseThreadpool 将其销毁.在调用这个函数后,我们将无法在将任何新的项添加到线程池的队列中.线程池中当前正在处理的队列中的线程会完成它们的处理并终止.此外,线程池的队列中所有尚未开始处理的项将被取消.
一旦我们创建了自己的线程池,并制定了线程池的最小数量和最大数量,我们就可以初始化一个回调环境,它包含了一些课应用于工作项的额外的设置或配置.(线程池回调环境结构 _TP_CALLBACK_ENVIRON ,其定义在 WinNT.h )
/*************************************** 显示此结果的编译环境为 VS2010 ***************************************/ #if (_WIN32_WINNT >= _WIN32_WINNT_WIN7) typedef struct _TP_CALLBACK_ENVIRON_V3 { TP_VERSION Version; PTP_POOL Pool; PTP_CLEANUP_GROUP CleanupGroup; PTP_CLEANUP_GROUP_CANCEL_CALLBACK CleanupGroupCancelCallback; PVOID RaceDll; struct _ACTIVATION_CONTEXT *ActivationContext; PTP_SIMPLE_CALLBACK FinalizationCallback; union { DWORD Flags; struct { DWORD LongFunction : 1; DWORD Persistent : 1; DWORD Private : 30; } s; } u; TP_CALLBACK_PRIORITY CallbackPriority; DWORD Size; } TP_CALLBACK_ENVIRON_V3; typedef TP_CALLBACK_ENVIRON_V3 TP_CALLBACK_ENVIRON, *PTP_CALLBACK_ENVIRON; #else typedef struct _TP_CALLBACK_ENVIRON_V1 { TP_VERSION Version; PTP_POOL Pool; PTP_CLEANUP_GROUP CleanupGroup; PTP_CLEANUP_GROUP_CANCEL_CALLBACK CleanupGroupCancelCallback; PVOID RaceDll; struct _ACTIVATION_CONTEXT *ActivationContext; PTP_SIMPLE_CALLBACK FinalizationCallback; union { DWORD Flags; struct { DWORD LongFunction : 1; DWORD Persistent : 1; DWORD Private : 30; } s; } u; } TP_CALLBACK_ENVIRON_V1; typedef TP_CALLBACK_ENVIRON_V1 TP_CALLBACK_ENVIRON, *PTP_CALLBACK_ENVIRON; #endif
然后我们可以调用 InitializeThreadpoolEnvironment 初始化这个结构体,接着当然回调环境必须调用 SetThreadpoolCallbackPool 标明给工作项应该由哪个线程池来处理.最后当我们不在需要这个使用线程池回调环境的时候,应该调用 DestroyThreadpoolEnvironment 来对它进行清理工作.
线程池的销毁(清理组)
为了得体地销毁私有线程池,我们首先可以需要通过调用 CreateThreadpoolCleanupGroup 来创建一个清理组,然后再将这个清理组与一个以绑定到线程池的回调函数结构体 TP_CALLBACK_ENVIRON 调用 SetThreadpoolCallbackCleanupGroup 函数把两者关联起来.其中 SetThreadpoolCallbackCleanupGroup 的 pfng 参数标识一个回调函数的地址(函数原型 即CleanupGroupCancelCallback),如果传给 pfng 参数值不为 NULL ,且当清理组被取消时那么这个回调函数会被调用.
当我们调用 CreateThreadpoolWork, CreateThreadpoolTimer, CreateThreadpoolWait 或 CreateThreadpoolIo 的时候,如果最后那个参数,即指向 PTP_CALLBACK_ENVIRON 结构体指针,不等于 NULL,那么所创建的项会被添加到对应的回调环境的清理组中,其目的是为了表示有线程池中添加了一项,需要潜在清理.在这些对了项完成后,如果我们调用 CloseThreadpoolWork, CloseThreadpoolTimer, CloseThreadpoolWait 和 CloseThreadpoolIo, 那就等于是隐式将对应的项从组中移除.
最后,在程序想要销毁线程池的时候,调用 CloseThreadpoolCleanupGroupMembers.这个函数与下面的 WaitForThreadpool*(例: WaitForThreadpoolWork)函数相似.当线程调用 CloseThreadpoolCleanupGroupMembers 时候,函数会一直等待,知道线程池的工作组中所有剩余的项(即已经创建当尚未关闭的项)都已经处理完毕为止.调用者还可以传 TRUE 给 fCancelPendingCallbacks 参数.这样会将所有已提交但尚未处理的工作项直接取消,函数会在所有当前正在运行的工作项王城之后返回.如果传给 fCancelPendingCallbacks 参数为 TRUE,而且传给 SetThreadpoolCallbackCleanupGroup 的 pfng 参数值是一个 CleanupGroupCancelCallback 函数地址,那么对每一个被取消的工作项,我们的回调函数会被调用,在 CleanupGroupCancelCallback 函数中.参数 ObjectContext 会包含每个被取消的上下文.(该上下文信息是通过 CreateThreadpool* 函数的 pv 参数设置的)在 CleanupGroupCancelCallback 函数中, 参数 CleanupContext 包含的上下文是通过 CloseThreadpoolCleanupGroupMembers 函数的 pvCleanupContext 参数传入的.如果在调用 CloseThreadpoolCleanupGroupMembers 时传入 FALSE 给 fCancelPendingCallbacks 参数.那么在返回之前,线程池会发时间来处理队列中所有剩余的项.注意这种情况下我们的 CleanupGroupCancelCallback 函数绝对不会被调用.因此可以传 NULL 给 pvCleanupContext 参数.
线程池制定与销毁代码样例
编写步骤
程序源码
#include#include #include using namespace std; VOID CALLBACK SimpleCallback(PTP_CALLBACK_INSTANCE Instance,PVOID Context); void main() { PTP_POOL tPool; TP_CALLBACK_ENVIRON pcbe; //创建线程池 tPool = CreateThreadpool(NULL); //设置线程池最大最小的线程数量 SetThreadpoolThreadMinimum(tPool,1); SetThreadpoolThreadMaximum(tPool,2); //初始化线程池环境变量 InitializeThreadpoolEnvironment(&pcbe); //为线程池设置线程池环境变量 SetThreadpoolCallbackPool(&pcbe,tPool); //单次工作提交 TrySubmitThreadpoolCallback(SimpleCallback,NULL,&pcbe); system("pause"); //清理线程池的环境变量 DestroyThreadpoolEnvironment(&pcbe); //关闭线程池 CloseThreadpool(tPool); } VOID CALLBACK SimpleCallback(PTP_CALLBACK_INSTANCE Instance,PVOID Context) { cout<<"this is SimpleCallback function!"<<endl; }
运行结果
以异步的方式调用函数
相关函数
编写步骤
代码样例
程序源码
运行结果
以时间段来调用函数
相关函数
编写步骤
代码样例
程序源码
#include#include #include using namespace std; VOID CALLBACK TimerCallback(PTP_CALLBACK_INSTANCE Instance,PVOID Context,PTP_TIMER Timer); void main() { PTP_TIMER tpTimer; FILETIME liDueTime; LARGE_INTEGER liUTC; liUTC.QuadPart = -30000000; liDueTime.dwLowDateTime = liUTC.LowPart; liDueTime.dwHighDateTime = liUTC.HighPart; //创建定时调用的工作对象 tpTimer = CreateThreadpoolTimer(TimerCallback,NULL,NULL); //判断定时调用的工作对象是否注册过计时器 if(!IsThreadpoolTimerSet(tpTimer)) { //为定时调用工作对象注册计时器 SetThreadpoolTimer(tpTimer,&liDueTime,0,0); } //睡眠主进程,等待计时器添加工作 Sleep(4000); //等待定时调用的工作对象 WaitForThreadpoolTimerCallbacks(tpTimer,false); //关闭定时调用的工作对象 CloseThreadpoolTimer(tpTimer); system("pause"); } VOID CALLBACK TimerCallback(PTP_CALLBACK_INSTANCE Instance,PVOID Context,PTP_TIMER Timer) { cout<<"this is TimerCallback function!"<<endl; }
运行结果
以内核对象的触发状态来调用函数
运行原理
其实,在此功能中线程池在内部会让一个线程调用 WaitForMultipleObjects 函数,传入通过 SetThreadpoolWait 函数注册一组句柄,并传入 false 给 bWaitAll 参数,这样当任何一个句柄被触发的时候,线程池就会被唤醒.由于 WaitForMultipleObjects 有一个限制,一次嘴甜只能等待 64(MAXMUM_WAIT_OBJECTS)个句柄,因此线程池事实上正是为每 64 个内核对象分配一个线程来进行等待,其效率还是相当高的.另外,由于 WaitForMultipleObjects 不允许我们将同一个句柄传入多次,因此我们必须确保不会用 SetThreadpoolWait 来多次注册同一个句柄.但是,我们可以调用 DuplicateHandle, 这样就可以分别注册原始句柄和复制句柄.
注意
绝对不要让回调函数调用 WaitForThreadpoolWait 并将自己的工作项作为参数传入,因为这样会导致死锁.另外,当线程在等待传给 SetThreadpoolWait 的句柄时,我们应该确保该句柄不会被关闭.最后我们可能并不想通过 PulseEvent 来触发一个已注册的事件,因为当 PulseEvent 被调用的时候,我们无法保证线程池正好在等待该事件.
相关函数
编写步骤
代码样例
程序源码
#include#include #include using namespace std; VOID CALLBACK WaitCallback(PTP_CALLBACK_INSTANCE Instance,PVOID Context,PTP_WAIT Wait,TP_WAIT_RESULT WaitResult); void main() { PTP_WAIT tpWait; HANDLE hMutex; //创建等待内核对象触发的工作对象 tpWait = CreateThreadpoolWait(WaitCallback,NULL,NULL); //创建互斥对象 hMutex=CreateMutex(NULL,false,0); //等待互斥对象被释放后获得互斥对象拥有权 WaitForSingleObject(hMutex,INFINITE); //设置等待内核事件触发调用回调函数 SetThreadpoolWait(tpWait,hMutex,0); //释放互斥对象拥有权 ReleaseMutex(hMutex); //等待等待内核对象触发的工作对象结束 WaitForThreadpoolWaitCallbacks(tpWait,false); //关闭等待内核对象触发的工作对象 CloseThreadpoolWait(tpWait); system("pause"); } VOID CALLBACK WaitCallback(PTP_CALLBACK_INSTANCE Instance,PVOID Context,PTP_WAIT Wait,TP_WAIT_RESULT WaitResult) { cout<<"this is WaitCallback function!"<<endl; }
运行结果
以异步 I/O 请求完成时调用函数
相关函数
代码样例
程序源码
运行结果
对回调函数的操作
回调函数的终止操作
线程池提供了一种便利的方法,用来描述在我们的回调函数返回之后,应该执行的一些操作,回调函数用传给它的不透明的 Instance 参数(其类型为 PTP_CALLBACK_INSTANCE)来调用下面的函数
另外两个函数