[笔记]Windows核心编程《十一》Windows线程池

简介: Windows核心编程《十一》Windows线程池

系列文章目录



[笔记]Windows核心编程《一》错误处理、字符编码

[笔记]Windows核心编程《二》内核对象

[笔记]Windows核心编程《三》进程

[笔记]Windows核心编程《四》作业

[笔记]快乐的LInux命令行《五》什么是shell

[笔记]Windows核心编程《五》线程基础

[笔记]Windows核心编程《六》线程调度、优先级和关联性

[笔记]Windows核心编程《七》用户模式下的线程同步

[笔记]Windows核心编程《八》用内核对象进行线程同步

[笔记]Windows核心编程《九》同步设备I/O和异步设备I/O

[笔记]Windows核心编程《十一》Windows线程池

[笔记]Windows核心编程《十二》纤程

[笔记]Windows核心编程《十三》windows内存体系结构

[笔记]Windows核心编程《十四》探索虚拟内存

[笔记]Windows核心编程《十五》在应用程序中使用虚拟内存

[笔记]Windows核心编程《十六》线程栈

[笔记]Windows核心编程《十七》内存映射文件

[笔记]Windows核心编程《十八》堆栈

[笔记]Windows核心编程《十九》DLL基础

[笔记]Windows核心编程《二十》DLL的高级操作技术

[笔记]Windows核心编程《二十一》线程本地存储器TLS

[笔记]Windows核心编程《二十二》注入DLL和拦截API

[笔记]Windows核心编程《二十三》结构化异常处理


文章目录



   系列文章目录

   前言

   11.1 情形1:以异步的方式调用函数

       11.1.1 显式地控制控制

           创建一个工作项

           向线程池提交一个请求

           取消已经提交的工作项或是等待工作项处理完毕

           关闭工作项

       11.1.2 Batch示例程序

   11.2 情形2:每隔一段时间调用一个函数

       11.2.1 线程池实现计时器

           通知线程池何时调用我们回调

           向线程池注册计时器(或者对计时器进行修改)

           确定某个计时器是否已经被设置

           等待一个计时器完成

       11.2.2 Timed Message Box示例程序

   11.3 情形3:在内核对象触发时调用一个函数

           内核对象被触发时执行某函数

           绑定内核对象到线程池

           获得WaitCallback被调用的原因

           等待一个等待项完成

           释放一个等待项的内存

   11.4 情形4: 在异步I/O 请求完成时调用一个函数

           定义回调函数

           创建一个线程池IO对象

           IO项中的设备与IO完成端口关联

           停止线程池调用回调函数

           取消设备与线程池的关联

           让另一个线程等待一个待处理的IO请求完成

   11.5 回调函数的终止操作

       11.5.1 对线程池进行定制

           创建新的线程池

           回调环境pcbe

       11.5.2 得体地销毁线程池:清理组

           创建一个清理组

           将清理组与一个已经绑定到线程池的TP_CALLBACK_ENVIRON结构关联

           销毁线程池

           释放清理组

   总结


前言



线程池通常含义指 一个固定数量的线程队列。每当需要一个线程去执行某任务(某段代码),从队列中选出一个闲置的线程去执行,当线程执行完某任务后,不会立即销毁,会回到队列中,等待执行其他任务。


为了简化程序员的工作,Windows提供了一个线程池机制来简化线程的创建、销毁以及日常管理。这个新线程池可能不适用于所有的情况,但大多数情况下它都能够满足我们的需要。

这个线程池能够帮助我们做一下事情:


   以异步的方式调用一个函数。

   每隔一段时间调用一个函数。

   当内核对象触发时调用一个函数。

   当异步IO请求完成时调用一个函数。


11.1 情形1:以异步的方式调用函数



为了让线程池来以异步地方式执行一个函数,我们需要 定义一个具有以下原型地函数:

VOID NTAPI SimpleCallback(
PTP_CALLBACK_INSTANCE pInstance,
PVOID pvContext
);

为了让线程池中的一个线程执行该函数,我们需要向线程池提交一个请求。

BOOL TrySubmitThreadpoolCallback(  
      PTP_SIMPLE_CALLBACK pfnCallback,  
      PVOID pvContext,  
      PTP_CALLBACK_ENVIRON pche);  

若调用成功,则返回true。否则返回false。


系统会自动为我们的进程创建一个默认的线程池,并让线程池中的一个线程来调用我们的回调函数。当这个线程处理完一个客户请求后,不会立即被销毁,而是会回到线程池,准备好处理队列中的其它工作项。线程池会不断重复使用其中的线程。如果线程池检测到创建另一个线程将能够更好地为应用程序服务,那它就会这样做。如线程池检测到它的线程数量已供过于求,那么它就会销毁其中一些。


11.1.1 显式地控制控制


在某些情况下,如内存不足时TrySubmitThreadpoolCallback可能会失败。每一次调用TrySubmitThreadpoolCallback时,系统会在内部分配一个工作项。

如果打算提交大量的工作项,出于性能和内存使用方面的考虑,应该手动创建工作项然后多次提交它。


创建一个工作项

VOID CALLBACK WorkCallback(  
        PTP_CALLBACK_INSTANCE Instance,  
        PVOID Context,  
        PTP_WORK Work);
PTP_WORK CreateThreadpoolWork(
    PTP_WORK_CALLBACK pfnWorkHandler,
    PVOID pvContext,
    PTP_CALLBACK_ENVIRON pcbe
);

pfnWorkHandler :函数指针,线程池中的线程最终对工作项处理时,会调用该函数指针指向的函数。

pvContext :传给回调函数的值。

pcbe : 如果传给它NULL则表示我们会将工作项添加到默认的线程池中。一般情况下默认的线程池能够满足大多数情况下的要求。


向线程池提交一个请求

VOID SubmitThreadpoolWork(PTP_WORK pWork); 


取消已经提交的工作项或是等待工作项处理完毕

如果我们想取消已经提交的工作项或是等待工作项处理完毕。可以调用以下函数:

VOID WaitForThreadpoolWorkCallbacks(  
      PTP_WORK pWork,  
      BOOL bCancelPendingCallbacks);  

此函数将线程挂起,直到工作项处理完毕。


pWork : 指向一个工作项。此工作项可以是CreateThreadpoolWork和SubmitThreadpoolWork来创建和提交的。如果工作项尚未被提交,那么等待函数立即返回。


bCancelPendingCallbacks:

TRUE,当指定的工作项尚未被处理,函数将其标记为取消并返回;当指定的工作项正在被处理,等待处理完成后,再返回;当同一工作项被提交多次时,只等待当前正处理的。

FALSE,当指定的工作项被处理完且处理它的线程已经被收回并准备处理下一个工作项时,才返回;当同一工作项被提交多次时,等待所有均被处理后,返回。


关闭工作项


当不需要一个工作项时,可以调用CloseThreadpoolWork,并传入指向该工作项的指针。

VOID CloseThreadpoolWork(PTP_WORK pwk);  

11.1.2 Batch示例程序


Batch示例程序,如何使用线程池的工作项。

待。


11.2 情形2:每隔一段时间调用一个函数



有时候应用程序需要在某些时间执行某些任务。Windows提供了可等待计时器对象,它使我们我们可以非常方便的得到一个时间通知。我们可以为每个需要执行基于时间的任务创建一个可等待的计时器对象,但这是不必要的。线程池函数为我们解决了这些事情。


11.2.1 线程池实现计时器

通知线程池何时调用我们回调


为了将一个工作项安排在某个时间执行,我们必须定义一个回调函数。该函数会在某个时刻被调用。回调函数原型为:

VOID CALLBACK TimeoutCallback(  
       PTP_CALLBACK_INSTANCE pInstance,  
       PVOID pvContext,  
       PTP_TIMER pTimer); 

然后调用下面的函数来通知线程池应在何时调用我们的函数:

PTP_TIMER CreateThreadpoolTimer(  
       PTP_TIMER_CALLBACK pfnTimerCallback,  
       PVOID pvContext,  
       PTP_CALLBACK_ENVIRON pcbe);  

这个函数与前面介绍的CreateThreadpoolWork相似。CreateThreadpoolTimer返回计时器对象。该计时器对象由CreateThreadpoolTimer函数创建并返回。

pfnTimerCallback :是一个函数指针。指向前面介绍的回调函数TimeroutCallback。每当线程池调用pfnTimerCallback指向的函数时会将pvContext传给它,并传给pTimer一个由CreateThreadpoolTimer返回的计时器对象指针。

pvContext:为传给回调函数参数。

pcbe:使用的线程池类型 同上,为NULL则是默认线程池。


向线程池注册计时器(或者对计时器进行修改)


当我们想要向线程池注册计时器时,应该调用SetThreadpoolTimer:

VOID SetThreadpoolTimer(  
       PTP_TIMER pTimer,  
       PFILETIME pftDueTime,  
       DWORD msPeriod,  
       DWORD msWindowLength);  

pTimer:用来标识CreateThreadpoolTimer返回的计时器对象。

pftDueTime:表示第一次调用回调函数是什么时候。传入一个负值表示一个相对时间。该时间相对于调用SetThreadpoolTimer的时间。传入-1表示立即调用。传入的正值以100ns为单位,从1600年的1月1日开始计算。

msPeriod:表示调用回调函数的时间间隔,传入0表示只调用1次。

msWindowLength:用来给回调函数的执行增加一些随机性。这使得回调函数会在当前设定的时间到当前设定的触发时间加上msWindowLength设定的时间之间触发。这对于多个计时器来说非常有用。这可以避免多个计时器间的冲突。


确定某个计时器是否已经被设置

BOOL IsThreadpoolTimerSet(PTP_TIMER pti);  


等待一个计时器完成


最后我们可以通过WaitForThreadpoolTimerCallbacks来等待一个计时器完成。调用CloseThreadpoolTimer来释放计时器的内存。它们与前面介绍的WaitForThreadpoolWork和CloseThreadpoolWork相似。


11.2.2 Timed Message Box示例程序


待完成。


11.3 情形3:在内核对象触发时调用一个函数



在实际使用中我们会发现我们会经常的等待一个内核对象被触发,触发后等待线程又会进入下一轮循环继续等待。Windows线程池提供了一些机制可以简化我们的工作。


内核对象被触发时执行某函数


如果我们想让内核对象被触发时执行某函数。需要进行以下步骤,

首先编写一个回调函数,它是内核对象被触发时被调用的函数。需要满足以下原型:

VOID CALLBACK WaitCallback(
    PTP_CALLBACK_INSTANCE pInstance,
    PVOID Context,
    PTP_WAIT Wait,
    TP_WAIT_RESULT WaitResult);

绑定内核对象到线程池

然后创建CreateThreadpoolWait来将一个内核对象绑定到线程池:

VOID SetThreadpoolWait(
   PTP_WAIT pWaitItem,
   HANDLE hObject,
   PFILETIME pftTimeout);

pWaitItem:用来标识CreateTheadpoolWait返回的对象。

hObject:用来标识内核对象。当此对象被触发时,回调函数会被调用。

pftTimeout:用来表示线程池最长应该花多少时间来等待内核对象被触发。传入0表示不用等待。传入负值表示相对时间。传NULL表示无限长的时间。


获得WaitCallback被调用的原因


线程池内部会让一个线程调用WaitForMultipleOBjecs函数。传入SetThreadpoolWait函数注册的一组句柄,并传入false给bWaitAll参数。当任何一个内核对象被触发时,线程池就会被唤醒。当内核对象被触发或是超出等待时间时,线程池的某个线程就会调用我们的回调函数(WaitCallback)。

WaitResult :用来表示WaitCallback被调用的原因。它可以是以下值:


   WAIT_OBJECT_0 超时之前有对象被触发。

   WAIT_TIMEOUT 由于超时导致回调函数被触发。

   WAIT_ABANDONED_0 如果传入的内核对象是互斥量且被遗弃。回调函数将收到这个值。


一旦线程池调用了我们的回调函数,对应的等待项将进入不活跃状态。所谓不活跃状态:如果想让回调函数在同一个内核对象被触发时再次被调用,我们需要调用SetThreadpoolWait来再次注册。


等待一个等待项完成


最后我们同样可以等待一个等待项完成。这可以调用:

WaitForThreadpoolWaitCallbacks。


释放一个等待项的内存

CloseThreadpoolWait

注意:不要让回调函数调用WaitForThreadpoolWork并将自己的工作项作为参数传入,这会导致死锁。


11.4 情形4: 在异步I/O 请求完成时调用一个函数



我们在上一篇博文中介绍了如何使用IO完成端口来高效的执行异步IO操作,也介绍了如何创建一个线程池并让其中的线程等待IO完成端口。


这里我们将介绍线程池如何管理线程的创建和销毁。在打开一个关联起来文件或设备时,我们必须现将该设备与线程池的IO完成端口,然后告诉线程池在异步IO完成时应该调用哪个函数。


定义回调函数


首先我们需要定义回调函数,它需要满足一下原型

VOID CALLBACK OverlappedCompletionRoutine(  
       PTP_CALLBACK_INSTANCE pInstance,  
       PVOID pvContext,  
       PVOID pOverlapped,  
       ULONG IoResult;  
       ULONG_PTR NumberOfBytesTransferred,  
       PTP_IO  pIo);

当一个IO操作完成时此回调函数会被调用并得到一个指向OVERLAPPED结构的指针。这个指针是我们在调用ReadFile或WriteFile来发出I/O请求时(通过pOverlapped参数)传入的。


IoResult:表示IO异步操作的执行结果。如果IO请求成功,将传给回调函数NO_ERROR。

NumberOfBytesTransferred :参数传入已传输的字节数。

pIo :传入指向线程池IO项的指针。马上介绍。

pInstance :后面会有介绍。


创建一个线程池IO对象


定义好回调函数后,我们就需要调用CreateThreadpoolIo来创建一个线程池IO对象。

PTP_IO CreateThreadpoolIo(  
       HANDLE hDevice,  
       PTP_WIN32_IO_CALLBACK pfnIoCallback,  
       PVOID pvContext,  
       PTP_CALLBACK_ENVIRON pcbe);  

hDevice :是与IO对象相关联的设备句柄。

pfnIoCallback :是前面我们介绍的回调函数指针。

pvContext : 当然是传给回调函数的参数。


IO项中的设备与IO完成端口关联


当IO对象创建好之后,我们就可以通过下面的函数来将嵌入在IO项中的设备与IO完成端口关联起来。

VOID StartThreadpoolIo(PTP_IO pio);  

关联之后我们就可以调用ReadFile或WriteFile了。此后当异步IO请求完成后,回调函数将会被调用。

注意,在每次调用ReadFile或WriteFile之前,我们必须调用StartThreadpoolIo,否则回调函数不会被调用。


停止线程池调用回调函数


此外我们还可以调用以下函数来停止线程池调用回调函数,此后回调函数将不会被调用:

VOID CancelThreadpoolIo(PTP_IO pio);  

取消设备与线程池的关联

CloseThreadpoolIo将取消设备与线程池的关联:

VOID CloseHandlepoolIo(PTP_IO pio);

让另一个线程等待一个待处理的IO请求完成

WaitForThreadpoolIoCallbacks函数让另一个线程等待一个待处理的IO请求完成。

VOID WaitForThreadpoolIoCallback(  
      PTP_IO pio,  
      BOOL bCancelPendingCallbacks);  


11.5 回调函数的终止操作


线程池提供了一种便利的方法,来描述我们的回调函数返回后,应该执行的一些操作。回调函数用传给它的不透明的pInstance来调用以下这些函数:

VOID LeaveCriticalSectionWhenCallbackReturns(
PTP_CALLBACK_INSTANCE pci,
PCRITICAL_SECTION pcs
);
VOID ReleaseMutexWhenCallbackReturns(
PTP_CALLBACK_INSTANCE pci,
HANDLE mut
);
VOID ReleaseSemaphoreWhenCallbackReturns(
PTP_CALLBACK_INSTANCE pci,
HANDLE sem,
DWORD crel
);
VOID SetEventWhenCallbackReturns(
PTP_CALLBACK_INSTANCE pci,
HANDLE evt
);
VOID FreeLibraryWhenCallbackReturns(
PTP_CALLBACK_INSTANCE pci,
HMODULE mod
);

11.5.1 对线程池进行定制


在调用CreateThreadpoolWork,CreateThreadpoolTimer,CreateThreadpoolWait或CreateThreadpoolIo时,有机会传入一个PTP_CALLBACK_ENVIRON参数。如传NULL,会将工作项添加到进程默认的线程池中。


创建新的线程池

PTP_POOL CreateThreadpool(
// 保留,传NULL。
PVOID reserved
);
// 设置线程池中线程最大,最小数量
BOOL SetThreadpoolThreadMinimum(
PTP_POOL pThreadPool,
DWORD cthrdMin
);
BOOL SetThreadpoolThreadMaximum(
PTP_POOL pThreadPool,
DWORD cthrdMost
);
// 关闭线程池。
// 线程池中线程结束本次处理后结束。
// 线程池队列中尚未开始处理的项将被取消。
VOID CloseThreadpool(PTP_POOL pThreadPool);

线程池始终保持池中的线程数量至少是最小数量,并允许线程数量增长到指定的最大数量。

回调环境pcbe

一旦我们创建了自己的线程池,并指定了线程的最小数量和最大数量,我们就可初始化一个回调环境,它包含了一些可用于工作项的额外的设置或配置。

线程池回调环境数据结构:

typedef struct _TP_CALLBACK_ENVIRON
{
  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 Private : 31;
    }s;
  } u;
}TP_CALLBACK_ENVIRON, *PTP_CALLBACK_ENVIRON;
// 初始化
VOID InitializeThreadpoolEnvironment(
PTP_CALLBACK_ENVIRON pcbe
);
// 清理
VOID DestroyThreadpoolEnvironment(
PTP_CALLBACK_ENVIRON pcbe
);
// 为了将一个工作项添加到线程池的队列中,回调环境必须标明该工作项应由那个线程池来处理。不调用时,pcbe初始化后的Pool字段为NULL,添加到进程默认的线程池。
VOID SetThreadpoolCallbackPool(
PTP_CALLBACK_ENVIRON pcbe,
PTP_POOL pThreadPool
);
// 确保,只要线程池中还有待处理的工作项,就将一个特定的DLL一直保持在进程的地址空间中。
VOID SetThreadpoolCallbackLibrary(
PTP_CALLBACK_ENVIRON pcbe,
PVOID mod
);
.

11.5.2 得体地销毁线程池:清理组


线程池可处理大量的队列项,这些项的来源各不相同。

这使我们很难知道线程池结束处理队列项的确切时间,但只有这样才能得体地将它销毁。

为了帮助我们对线程池进行得体的清理,线程池提供了清理组。

默认的线程池不会被销毁。在进程终止时,windows会将其销毁并负责所有的清理工作。


创建一个清理组

PTP_CLEANUP_GROUP CreateThreadpoolCleanupGroup();

将清理组与一个已经绑定到线程池的TP_CALLBACK_ENVIRON结构关联

VOID SetThreadpoolCallbackCleanupGroup(
  PTP_CALLBACK_ENVIRON pcbe,
  PTP_CLEANUP_GROUP ptpcg,
  // 清理组被取消时,回调函数会被调用
  PTP_CLEANUP_GROUP_CANCEL_CALLBACK pfng
);
VOID CALLBACK CleanupGroupCancelCallback(
  PVOID pvObjectContext,
  PVOID pvCleanupContext
);

这个函数在内部会设置PTP_CALLBACK_ENVIRON的CleanupGroup字段和CleanupGroupCancelCallback字段。


当调用CreateThreadpoolWork,CreateThreadpoolTimer,CreateThreadpoolWait,CreateThreadpoolIo时,如最后的的参数,即指向PTP_CALLBACK_ENVIRON结构的指针不为NULL,那么创建的项会被添加到对应的回调环境的清理组中。


在这些队列项完成后,如我们调用了CloseThreadpoolWork,CloseThreadpoolTimer,CloseThreadpoolWait和CloseThreadpoolIo,等于是隐式地将对应的项从清理组中移除。


销毁线程池

// 函数会一直等待,直到线程池的工作组中所有剩余的项(创建但未关闭的)都已处理完毕。
VOID CloseThreadpoolCleanupGroupMembers(
PTP_CLEANUP_GROUP ptpcg,
// TRUE,在所有当前正在运行的工作项完成后返回。尚未处理的取消。
// FALSE,等待所有项处理完毕
BOOL bCancelPendingCallbacks,
PVOID pvCleanupContext
);

如果传给bCancelPendingCallbacks值为TRUE,且传给SetThreadpoolCallbackCleanupGroup的pfng不为NULL。则,对每个被取消的工作项,pfng会被调用。回调函数的pvObjectContext包含每个被取消项的上下文(CreateThreadpoolxxx时设置的)。回调函数的pvCleanupContext是上述调用的最后一个参数。


释放清理组

VOID WINAPI CloseThreadpoolCleanupGroup(
PTP_CLEANUP_GROUP ptpcg
);
// 销毁自定义线程池的环境
DestroyThreadpoolEnvironment
// 关闭线程池
CloseThreadpool

总结


相关:

参考1

参考2

参考3

相关文章
|
2月前
|
Java 开发者
Java多线程编程中的常见误区与最佳实践####
本文深入剖析了Java多线程编程中开发者常遇到的几个典型误区,如对`start()`与`run()`方法的混淆使用、忽视线程安全问题、错误处理未同步的共享变量等,并针对这些问题提出了具体的解决方案和最佳实践。通过实例代码对比,直观展示了正确与错误的实现方式,旨在帮助读者构建更加健壮、高效的多线程应用程序。 ####
|
2月前
|
安全 Java UED
深入浅出Java多线程编程
【10月更文挑战第40天】在Java的世界中,多线程是提升应用性能和响应能力的关键。本文将通过浅显易懂的方式介绍Java中的多线程编程,从基础概念到高级特性,再到实际应用案例,带你一步步深入了解如何在Java中高效地使用多线程。文章不仅涵盖了理论知识,还提供了实用的代码示例,帮助你在实际开发中更好地应用多线程技术。
57 5
|
13天前
|
存储 安全 Java
Java多线程编程秘籍:各种方案一网打尽,不要错过!
Java 中实现多线程的方式主要有四种:继承 Thread 类、实现 Runnable 接口、实现 Callable 接口和使用线程池。每种方式各有优缺点,适用于不同的场景。继承 Thread 类最简单,实现 Runnable 接口更灵活,Callable 接口支持返回结果,线程池则便于管理和复用线程。实际应用中可根据需求选择合适的方式。此外,还介绍了多线程相关的常见面试问题及答案,涵盖线程概念、线程安全、线程池等知识点。
93 2
|
2月前
|
设计模式 Java 开发者
Java多线程编程的陷阱与解决方案####
本文深入探讨了Java多线程编程中常见的问题及其解决策略。通过分析竞态条件、死锁、活锁等典型场景,并结合代码示例和实用技巧,帮助开发者有效避免这些陷阱,提升并发程序的稳定性和性能。 ####
|
1月前
|
缓存 Java 调度
多线程编程核心:上下文切换深度解析
在现代计算机系统中,多线程编程已成为提高程序性能和响应速度的关键技术。然而,多线程编程中一个不可避免的概念就是上下文切换(Context Switching)。本文将深入探讨上下文切换的概念、原因、影响以及优化策略,帮助你在工作和学习中深入理解这一技术干货。
47 10
|
2月前
|
缓存 Java 开发者
Java多线程编程的陷阱与最佳实践####
本文深入探讨了Java多线程编程中常见的陷阱,如竞态条件、死锁和内存一致性错误,并提供了实用的避免策略。通过分析典型错误案例,本文旨在帮助开发者更好地理解和掌握多线程环境下的编程技巧,从而提升并发程序的稳定性和性能。 ####
|
29天前
|
安全 算法 Java
Java多线程编程中的陷阱与最佳实践####
本文探讨了Java多线程编程中常见的陷阱,并介绍了如何通过最佳实践来避免这些问题。我们将从基础概念入手,逐步深入到具体的代码示例,帮助开发者更好地理解和应用多线程技术。无论是初学者还是有经验的开发者,都能从中获得有价值的见解和建议。 ####
|
29天前
|
Java 调度
Java中的多线程编程与并发控制
本文深入探讨了Java编程语言中多线程编程的基础知识和并发控制机制。文章首先介绍了多线程的基本概念,包括线程的定义、生命周期以及在Java中创建和管理线程的方法。接着,详细讲解了Java提供的同步机制,如synchronized关键字、wait()和notify()方法等,以及如何通过这些机制实现线程间的协调与通信。最后,本文还讨论了一些常见的并发问题,例如死锁、竞态条件等,并提供了相应的解决策略。
50 3
|
1月前
|
算法 调度 开发者
多线程编程核心:上下文切换深度解析
在多线程编程中,上下文切换是一个至关重要的概念,它直接影响到程序的性能和响应速度。本文将深入探讨上下文切换的含义、原因、影响以及如何优化,帮助你在工作和学习中更好地理解和应用多线程技术。
42 4
|
10天前
|
安全 Java API
【JavaEE】多线程编程引入——认识Thread类
Thread类,Thread中的run方法,在编程中怎么调度多线程