JAVA线程池原理333
java 线程池工作原理
java 线程池工作原理一、概述Java线程池是Java多线程编程中非常重要的一个概念,它可以有效地管理和控制线程的数量,从而提高程序的性能和稳定性。
本文将介绍Java线程池的工作原理,包括线程池的基本构成、线程池如何工作以及常见的线程池类型等。
二、线程池的基本构成Java线程池由以下几个基本组成部分组成:1.任务队列:用于存储等待执行的任务。
2.工作线程:用于执行任务的线程。
3.任务提交接口:用于向线程池提交任务。
4.管理接口:用于管理线程池,例如增加或减少工作线程数量等。
三、Java 线程池如何工作当一个任务被提交到Java线程池时,如果当前工作线程数量小于核心线程数(corePoolSize),则会创建一个新的工作线程来执行该任务。
如果当前工作线程数量已经达到核心线数,则该任务会被放入任务队列中等待执行。
如果队列已满,则会创建新的非核心工作线去执行该任务(如果非核心工作数未达到最大值),否则将根据拒绝策略进行处理(例如抛出异常或直接丢弃该任务)。
当一个工作线程执行完一个任务后,它会从任务队列中获取下一个任务并继续执行。
如果工作线程在一定时间内没有获取到新的任务,则会被回收(如果超过了核心线程数)或保持等待状态(如果未超过核心线程数)。
四、常见的线程池类型Java线程池提供了几种不同的类型,以满足不同的需求:1.固定大小线程池:该类型的线程池具有固定数量的工作线程,当所有工作线程都处于忙碌状态时,新提交的任务将被放入队列中等待。
这种类型的线程池适用于执行长时间运行的任务。
2.缓存型线程池:该类型的线程池没有固定数量的工作线程,而是根据需要动态创建和销毁工作线程。
如果当前有可用的空闲工作线程,则新提交的任务将立即分配给它们;否则将创建新的工作线程来执行该任务。
这种类型的线程池适用于执行短时间运行的任务。
3.单一线程池:该类型只有一个工作线程,所有提交到该线程池中的任务都将按顺序依次执行。
这种类型适用于需要保证顺序执行多个任务时。
java线程池工作原理
java线程池工作原理Java线程池是一种有效的编程技术,它可以帮助我们节约时间,节省资源,提高应用程序执行效率。
通过使用线程池,可以重复利用许多线程,而不是每次都要创建新的线程,从而节约资源。
线程池使用一种称为“工作窃取”的技术来提高应用程序的执行效率。
Java线程池的工作原理如下:首先,当程序的执行任务到达时,它们将被添加到线程池的等待队列中,一旦有空闲的线程,它会立即从队列中抽取一个任务并开始执行。
当任务完成时,线程会被返回到线程池中,而不是被终止,以备下次使用。
当线程池中的线程都已被占用时,称之为“线程池过载”,此时,如果又有新的任务到达,线程池将把这些任务放到一个另外的队列中,这就是称之为“工作窃取”技术的原因。
在工作窃取技术中,空闲线程会从其他线程池中窃取任务,从而提高效率。
除此之外,Java线程池还包括一些其他功能,如控制池大小,终止线程池等,以确保程序的有效运行。
比如,可以配置线程池,设置其大小,以控制系统中并发线程的数量。
此外,还可以设置超时时间,告诉线程池在多长时间内没有新任务传入时,它将自动关闭或重新初始化已存在的线程。
Java线程池在提高应用程序性能、节省资源、提高任务执行效率等方面发挥着重要作用。
它能够自动重用可用线程,从而减少了程序的资源消耗,并且能够实现更高的性能、更快的响应速度。
此外,它还可以通过设置线程池大小、超时时间等参数,来控制程序执行效率,从而实现对系统的有效控制。
因此,Java线程池是Java程序开发者必须了解的一项技术,它能够帮助我们提高程序的执行效率,减少资源消耗,提高程序的可靠性。
线程池的优势在于,它能够自动的重用空闲的线程,以及通过工作窃取技术来实现更高的性能。
线程池的工作原理
线程池的工作原理线程池可以大大提高系统的效率,是一种非常流行的应用程序开发技术。
它可以有效地减少系统开销,提高系统性能,延长系统寿命。
在这篇文章中,我们将探讨线程池的工作原理,以及如何利用它来提高系统的性能。
线程池的基本概念是在运行时创建一组可以被多次重复使用的线程。
这些线程被存储在一个叫做线程池的容器中,以便在需要执行任务时可以被多次重复使用。
当需要执行一个任务时,系统会从线程池中取出一个线程并将它分配给该任务,然后将该线程返回到线程池中,以便在未来的某个时间再被重复使用。
首先,在使用线程池之前,必须创建和初始化一个线程池,以使其能够正常工作。
一般来说,线程池有两种类型:阻塞线程池和非阻塞线程池。
在阻塞线程池中,线程池中的线程都是阻塞状态,这意味着在分配线程执行任务时必须要等待池中某个线程释放,然后才能够分配给新任务。
而在非阻塞线程池中,则可以立即分配给新任务。
一旦线程池被创建和初始化,它将开始接受任务并运行这些任务。
当一个任务被分配到线程池中的线程时,线程会尝试执行该任务。
如果任务失败(可能因为系统超时或其他错误),线程会将任务返回到线程池,以便以后再次尝试。
当任务执行结束时,线程会将该任务从线程池中移除,并将该线程返回到线程池。
线程池具有很多优点,最显著的一个优点就是可以有效地把一个大量的任务分解为小任务,然后分配给多个线程运行,从而加快系统的响应时间。
此外,线程池还可以提高程序的可维护性和可扩展性,因为线程池中的线程都是可以重复使用的,因此可以在添加新线程时减少开销,从而提高应用程序的性能或效率。
综上所述,线程池的工作原理是:首先创建和初始化一个线程池,然后运行这些任务;接着,将任务分配给线程池中的线程,以及当任务结束时将该线程返回到线程池中。
线程池有效地将大量的任务分解为小任务,提高了系统的性能,更加可靠可维护,更加可扩展性。
它可以大大提高系统的性能,延长系统的有效寿命,减少系统开销。
Java线程池工作原理
Java线程池⼯作原理前⾔当项⽬中有频繁创建线程的场景时,往往会⽤到线程池来提⾼效率。
所以,线程池在项⽬开发过程中的出场率是很⾼的。
那线程池是怎么⼯作的呢?它什么时候创建线程对象,如何保证线程安全...什么时候创建线程对象当实例化线程池对象时,并没有预先创建corePoolSize个线程对象,⽽是在调⽤execute或submit提交任务时,才会创建线程对象。
⼯作流程public void execute(Runnable command) {int c = ctl.get();// 1. 如果核⼼线程数没有⽤完,则让核⼼线程执⾏任务if (workerCountOf(c) < corePoolSize) {if (addWorker(command, true))return;c = ctl.get();}// 2. 核⼼线程⽤完后,尝试添加到等待队列if (isRunning(c) && workQueue.offer(command)) {int recheck = ctl.get();if (! isRunning(recheck) && remove(command))reject(command);else if (workerCountOf(recheck) == 0)addWorker(null, false);}// 3. 等待队列满后,尝试创建临时线程执⾏任务;// 如果创建临时线程失败,即达到了最⼤线程数,则采⽤拒绝策略处理else if (!addWorker(command, false))reject(command);}如何保存线程池状态// 线程池采⽤状态压缩的思想,通过 32 个 bit 位来存储线程池状态和⼯作线程数量// 其中,⾼ 3 位存储线程池状态,低 29 位存储⼯作线程数量// 表⽰⼯作线程数量位数private static final int COUNT_BITS = Integer.SIZE - 3;// 线程池最⼤容量private static final int CAPACITY = (1 << COUNT_BITS) - 1;// 通过线程安全的 atomic 对象来存储线程池状态private final AtomicInteger ctl = new AtomicInteger(ctlOf(RUNNING, 0));private static int ctlOf(int rs, int wc) { return rs | wc; }如何保证线程安全1、对于⼀些需要线程共享的成员变量,通过volatile修饰2、线程池状态和⼯作线程数的修改通过AtomicInteger类⾃带的线程安全⽅法实现3、⼯作线程Worker通过线程不安全的HashSet存储,每次操作HashSet时都通过⼀个可重⼊锁ReentrantLock保证线程安全,ReentrantLock还⽤来保证访问⼀些线程不安全的变量,⽐如/*** Tracks largest attained pool size. Accessed only under* mainLock.*/private int largestPoolSize;/*** Counter for completed tasks. Updated only on termination of* worker threads. Accessed only under mainLock.*/private long completedTaskCount;4、Worker实现了AQS类,保证线程安全地操作⼯作线程Worker。
java多线程:线程池原理、阻塞队列
java多线程:线程池原理、阻塞队列⼀、线程池定义和使⽤jdk 1.5 之后就引⼊了线程池。
1.1 定义从上⾯的空间切换看得出来,线程是稀缺资源,它的创建与销毁是⼀个相对偏重且耗资源的操作,⽽Java线程依赖于内核线程,创建线程需要进⾏操作系统状态切换。
为避免资源过度消耗需要设法重⽤线程执⾏多个任务。
线程池就是⼀个线程缓存,负责对线程进⾏统⼀分配、调优与监控。
(数据库连接池也是⼀样的道理)什么时候使⽤线程池?单个任务处理时间⽐较短;需要处理的任务数量很⼤。
线程池优势?重⽤存在的线程,减少线程创建、消亡的开销,提⾼性能、提⾼响应速度。
当任务到达时,任务可以不需要等到线程创建就能⽴即执⾏。
提⾼线程的可管理性,可统⼀分配,调优和监控。
1.2 线程池在 jdk 已有的实现在 juc 包下,有⼀个接⼝:Executor :Executor ⼜有两个⼦接⼝:ExecutorService 和 ScheduledExecutorService,常⽤的接⼝是 ExecutorService。
同时常⽤的线程池的⼯具类叫 Executors。
例如:ExecutorService service = Executors.newCachedThreadPool();Executor 框架虽然提供了如 newFixedThreadPool()、newSingleThreadExecutor()、newCachedThreadPool()、newScheduledThreadPool() 等创建线程池的⽅法,但都有其局限性,不够灵活。
上⾯的⼏种⽅式点进去会发现,都是⽤ ThreadPoolExecutor 进⾏创建的:newSingleThreadExecutor 字⾯意思简单线程执⾏器。
newFixedThreadPool 字⾯意思固定的线程池,传参就是线程固定数⽬,适⽤于执⾏长期任务的场景。
newCachedThreadPool 字⾯意思缓存线程池,核⼼线程0,最⼤线程⾮常⼤,动态创建的特点。
JAVA线程池原理以及几种线程池类型介绍
在什么情况下使用线程池?1.单个任务处理的时间比较短2.将需处理的任务的数量大使用线程池的好处:1.减少在创建和销毁线程上所花的时间以及系统资源的开销2.如不使用线程池,有可能造成系统创建大量线程而导致消耗完系统内存以及”过度切换”。
线程池工作原理:线程池为线程生命周期开销问题和资源不足问题提供了解决方案。
通过对多个任务重用线程,线程创建的开销被分摊到了多个任务上。
其好处是,因为在请求到达时线程已经存在,所以无意中也消除了线程创建所带来的延迟。
这样,就可以立即为请求服务,使应用程序响应更快。
而且,通过适当地调整线程池中的线程数目,也就是当请求的数目超过某个阈值时,就强制其它任何新到的请求一直等待,直到获得一个线程来处理为止,从而可以防止资源不足。
线程池的替代方案线程池远不是服务器应用程序内使用多线程的唯一方法。
如同上面所提到的,有时,为每个新任务生成一个新线程是十分明智的。
然而,如果任务创建过于频繁而任务的平均处理时间过短,那么为每个任务生成一个新线程将会导致性能问题。
另一个常见的线程模型是为某一类型的任务分配一个后台线程与任务队列。
AWT 和 Swing 就使用这个模型,在这个模型中有一个 GUI 事件线程,导致用户界面发生变化的所有工作都必须在该线程中执行。
然而,由于只有一个 AWT 线程,因此要在 AWT 线程中执行任务可能要花费相当长时间才能完成,这是不可取的。
因此,Swing 应用程序经常需要额外的工作线程,用于运行时间很长的、同 UI 有关的任务。
每个任务对应一个线程方法和单个后台线程(single-background-thread)方法在某些情形下都工作得非常理想。
每个任务一个线程方法在只有少量运行时间很长的任务时工作得十分好。
而只要调度可预见性不是很重要,则单个后台线程方法就工作得十分好,如低优先级后台任务就是这种情况。
然而,大多数服务器应用程序都是面向处理大量的短期任务或子任务,因此往往希望具有一种能够以低开销有效地处理这些任务的机制以及一些资源管理和定时可预见性的措施。
java线程池底层原理
java线程池底层原理Java线程池是一种用于管理和复用线程的机制,它可以提高程序的性能和资源利用率。
线程池的底层原理涉及到以下几个方面:1. 线程池的创建和初始化,在Java中,我们可以使用ThreadPoolExecutor类来创建线程池。
线程池的初始化包括设置核心线程数、最大线程数、线程存活时间、任务队列等参数。
2. 线程池的任务调度,线程池会维护一个任务队列,当有新的任务提交时,线程池会根据一定的策略从任务队列中取出任务,并将其分配给空闲的线程执行。
常见的任务调度策略有,直接执行、创建新线程、队列和拒绝策略。
3. 线程池的线程管理,线程池会管理线程的创建、销毁和复用。
当线程池中的线程数量小于核心线程数时,会创建新的线程。
当线程池中的线程数量达到核心线程数后,新的任务会被放入任务队列中等待执行。
当任务队列已满且线程池中的线程数量小于最大线程数时,会创建新的线程来执行任务。
当线程池中的线程数量超过最大线程数时,多余的线程会被销毁。
4. 线程池的线程执行,线程池中的线程会执行提交的任务。
执行任务的过程包括线程的启动、任务的执行和任务的完成。
线程执行完任务后,会返回线程池并等待下一个任务的分配。
5. 线程池的状态管理,线程池会维护一些状态信息,如线程池是否运行、线程池中的线程数量、任务队列的大小等。
这些状态信息可以帮助我们监控和管理线程池的运行情况。
总结起来,Java线程池的底层原理主要涉及到线程池的创建和初始化、任务调度、线程管理、线程执行和状态管理等方面。
通过合理的配置和使用线程池,我们可以更好地管理和复用线程,提高程序的性能和资源利用率。
java线程池工作原理及拒绝策略详解
java线程池⼯作原理及拒绝策略详解在多线程编程中,我们经常使⽤线程池来管理线程,以减缓线程频繁的创建和销毁带来的资源的浪费,在创建线程池的时候,经常使⽤⼀个⼯⼚类来创建线程池Executors,实际上Executors的内部使⽤的是类ThreadPoolExecutor。
它有⼀个最终的构造函数如下:corePoolSize:线程池中的核⼼线程数量,即使这些线程没有任务⼲,也不会将其销毁。
maximumPoolSize:线程池中的最多能够创建的线程数量。
keepAliveTime:当线程池中的线程数量⼤于corePoolSize时,多余的线程等待新任务的最长时间。
unit:keepAliveTime的时间单位。
workQueue:在线程池中的线程还没有还得及执⾏任务之前,保存任务的队列(当线程池中的线程都有任务在执⾏的时候,仍然有任务不断的提交过来,这些任务保存在workQueue队列中)。
threadFactory:创建线程池中线程的⼯⼚。
handler:当线程池中没有多余的线程来执⾏任务,并且保存任务的多列也满了(指的是有界队列),对仍在提交给线程池的任务的处理策略。
RejectedExecutionHandler 是⼀个策略接⼝,⽤在当线程池中没有多余的线程来执⾏任务,并且保存任务的多列也满了(指的是有界队列),对仍在提交给线程池的任务的处理策略。
公共策略接⼝:这个策略接⼝有四个实现类:AbortPolicy:该策略是直接将提交的任务抛弃掉,并抛出RejectedExecutionException异常。
DiscardPolicy:该策略也是将任务抛弃掉(对于提交的任务不管不问,什么也不做),不过并不抛出异常。
DiscardOldestPolicy:该策略是当执⾏器未关闭时,从任务队列workQueue中取出第⼀个任务,并抛弃这第⼀个任务,进⽽有空间存储刚刚提交的任务。
使⽤该策略要特别⼩⼼,因为它会直接抛弃之前的任务。
java线程池的工作原理
java线程池的工作原理
Java线程池的工作原理如下:
1. 线程池的初始化:在使用线程池之前,需要首先创建一个线程池对象,并设定线程池的核心线程数、最大线程数、线程空闲时间等参数。
2. 任务提交:当有任务需要执行时,可以使用线程池的
submit()或execute()方法将任务提交给线程池。
3. 任务队列:线程池会维护一个任务队列,用于存储提交的任务。
如果线程池中的线程数没有达到核心线程数,线程池就会创建新的线程来执行任务。
如果线程池中的线程数已经达到核心线程数,但任务队列仍然可以存储新任务,线程池会将新任务存储在任务队列中。
4. 线程池的工作方式:线程池会不断地从任务队列中取出任务,并通过线程池中的线程来执行任务。
若线程池中的线程处于空闲状态,则会被重新利用,否则任务会等待直到有线程可用。
5. 线程池的扩容:当任务队列已满且线程池中的线程数未达到最大线程数时,线程池会创建新的线程来执行任务。
一旦线程数达到最大线程数,线程池将不再接受新的任务。
6. 线程池的关闭:当不再需要线程池时,可以调用线程池的shutdown()方法来关闭线程池。
关闭线程池后,线程池将不再
接受新的任务,同时会等待已提交的任务执行完毕。
可以使用
awaitTermination()方法来等待所有任务执行完毕。
线程池的好处是提高了线程的利用率,避免了频繁创建和销毁线程的开销。
同时可以控制线程的并发数,防止系统资源过度消耗。
线程池的运行原理
线程池的运行原理什么是线程池?线程池是一种用于管理和复用线程的机制,它可以提高多线程应用程序的性能和可管理性。
线程池维护一组预先创建的工作线程,这些工作线程可以被重复使用来执行任务。
线程池的基本原理1.创建线程池:首先,需要创建一个包含固定数量的工作线程的线程池。
这个固定数量通常根据系统资源和应用程序需求进行调整。
2.接收任务:当有新的任务需要执行时,应用程序将任务提交给线程池。
3.任务队列:线程池维护一个任务队列,用于存储待执行的任务。
当有新任务提交时,会被添加到任务队列中。
4.选择空闲线程:当有空闲的工作线程可用时,从任务队列中选择一个待执行的任务,并将其分配给该工作线程。
5.执行任务:被选中的工作线程会从任务队列中获取待执行的任务,并在自己的上下文中执行该任务。
6.完成任务:当工作线程完成了一个任务后,会返回结果并等待下一个任务分配。
线程池的好处•降低资源消耗:由于创建和销毁线程需要消耗系统资源,线程池可以复用线程,减少了线程创建和销毁的开销。
•提高响应速度:线程池可以避免频繁地创建和销毁线程,从而减少了响应请求的时间。
•提高系统稳定性:通过限制同时执行的任务数量,线程池可以防止系统超载,保持系统稳定运行。
•提供更好的管理机制:线程池提供了对工作线程的管理机制,例如增加或减少工作线程的数量、设置任务队列大小等。
线程池的组成部分1.工作线程:执行实际任务的线程。
在初始化时创建,并一直存在于整个生命周期中。
2.任务队列:用于存储待执行的任务。
通常是一个有界队列,当任务队列已满时,新提交的任务会被拒绝执行。
3.任务接口:定义了待执行任务所需要实现的接口或抽象类。
通过实现该接口或继承该抽象类来定义具体的任务。
4.线程池管理器:负责创建和管理整个线程池。
它包括创建工作线程、添加任务到队列、选择空闲线程等功能。
线程池的工作流程1.初始化:在初始化阶段,根据应用程序的需求创建线程池,并初始化任务队列。
2.提交任务:当有新的任务需要执行时,应用程序将任务提交给线程池。
java 线程池运行原理
java 线程池运行原理Java线程池是Java多线程编程中常用的一种机制,它能够管理和复用线程,提高多线程程序的性能和效率。
本文将从线程池的原理、工作机制和使用方法等方面介绍Java线程池的运行原理。
一、线程池的原理线程池的原理是将任务的执行和线程的创建、销毁分离开来,通过线程池来管理线程的生命周期。
线程池中包含一组工作线程,它们可用来执行提交的任务。
线程池的核心思想是将任务提交给线程池,线程池管理线程的创建、销毁以及任务的调度和执行。
二、线程池的工作机制1. 线程池的创建在Java中,可以通过ThreadPoolExecutor类来创建线程池。
ThreadPoolExecutor类是ExecutorService接口的一个实现类,它提供了一系列的构造方法和配置参数,可以根据实际需求来创建线程池。
2. 线程池的任务队列线程池内部有一个任务队列,用于存放提交的任务。
当有新的任务提交时,线程池将任务放入任务队列中等待执行。
3. 线程池的工作线程线程池中有一组工作线程,它们负责执行任务队列中的任务。
线程池会根据实际情况动态地创建或销毁工作线程,以满足任务的执行需求。
4. 线程池的任务调度线程池根据任务队列中的任务数量和线程池的配置参数,决定是否创建新的工作线程来执行任务。
如果任务队列中的任务数量超过了线程池的最大容量,线程池将拒绝接受新的任务。
5. 线程池的任务执行线程池中的工作线程会不断地从任务队列中取出任务并执行。
当任务执行完毕后,工作线程可以继续从任务队列中获取新的任务。
三、线程池的使用方法1. 创建线程池可以通过ThreadPoolExecutor类的构造方法来创建线程池,可以设置线程池的核心线程数、最大线程数、任务队列和拒绝策略等参数。
2. 提交任务可以通过线程池的submit()方法来提交任务,任务可以是Runnable接口或Callable接口的实现类。
3. 关闭线程池当不再需要线程池时,可以通过调用线程池的shutdown()方法来关闭线程池。
线程池 工作原理
线程池工作原理
线程池是一种并发编程的技术,它可以通过预先创建一定数量的线程,来处理多个任务,从而提高程序的运行效率。
线程池中包含若干个线程,这些线程可以执行多个任务。
线程池分为两类:定长线程池和可变长度线程池。
线程池的工作原理如下:
1. 线程池初始化
在初始化时,会创建一个线程池管理器,创建固定数量的线程池,将这些线程放在一个线程池数组中。
同时,创建一个任务队列,用来存储等待执行的任务。
2. 等待执行任务
当有任务到来时,线程池管理器从任务队列中取出一个任务,分配给其中一个空闲线程去执行。
如果此时线程池中的所有线程都在执行任务,则任务会被暂时缓存,等待有空闲的线程可以执行该任务。
3. 执行任务
线程池中的线程会竞争任务,并执行具体任务代码。
任务执行完毕之后,线程并不会结束生命周期,而是会进入等待状态,继续等待新的任务。
4. 结束任务
当线程池中的线程完成任务之后,如果没有新的任务需要执行,线程会进入休眠状态,等待新的任务的到来。
如果线程池不再需要使用,那么会将线程池中的线程全部终止。
线程池的作用是避免线程的频繁创建和销毁,相比于每次执行任务都创建和销毁线程,线程池可以大大降低线程创建和销毁的时间开销,提高程序运行效率。
java多线程的线程池实现原理
Java多线程的线程池实现原理一、概述在Java中,线程池是一种用于管理和复用线程的机制,可以有效地管理多线程的创建、执行和销毁。
线程池可以减少线程的创建和销毁的开销,提高线程的复用率,有效地提高了多线程程序的性能和稳定性。
本文将介绍Java中线程池的实现原理,以帮助读者更好地理解和使用线程池。
二、线程池的基本原理1. 线程池的基本结构线程池一般由三个主要组件组成,分别是线程池管理器、工作队列和线程工厂。
线程池管理器负责管理线程池的创建、销毁和执行,工作队列用于存储待执行的任务,线程工厂用于创建新的线程。
2. 线程池的工作流程当一个任务需要执行时,线程池管理器会首先从工作队列中取出一个空闲线程来执行任务。
如果工作队列中没有空闲线程,则线程池管理器会根据需要创建新的线程。
当一个线程执行完任务后,它会继续从工作队列中取出下一个任务并执行,直到工作队列为空。
如果工作队列中没有任务,空闲线程会等待一段时间后自动销毁,以节省系统资源。
三、线程池的实现原理1. 线程池的创建Java中线程池的创建通常使用Executors类提供的静态工厂方法来实现。
Executors类提供了一系列的静态方法,用于创建不同类型的线程池。
比较常用的有newFixedThreadPool、newCachedThreadPool 和newSingleThreadExecutor等方法。
2. 线程池的参数设置在创建线程池时,可以设置多种参数来控制线程池的行为。
比较常用的参数包括核心线程数、最大线程数、工作队列类型、线程超时时间等。
这些参数可以根据具体的应用场景进行调优,以达到最佳的性能和资源利用率。
3. 线程池的执行策略线程池中线程的执行策略一般有两种,分别是任务执行和线程执行。
任务执行策略是指线程池将任务分配给空闲线程执行,线程执行策略是指线程池每个线程执行完任务后再次从工作队列中取出新的任务执行。
Java中的线程池通常使用线程执行策略,以提高线程的复用率和系统的稳定性。
java 线程池原理
java 线程池原理
线程池是一种有效地解决系统性能问题和负责管理多个线程的机制,通常使用线程池来提高系统性能,减少 cpu 负载。
线程池原理,核心思想是维持一定数量的线程处于空闲状态,当
新任务加入时,由线程池中的某个空闲线程执行并完成任务,但这个
线程不会被销毁,而是处于空闲状态,等待下一次任务的到来。
处于
空闲状态的线程池总是会有一定数量,在某个特定情况下执行某个任
务不需要创建新线程,这样可以减少创建线程所带来的开销。
当线程池不能满足任务的提交要求时,可以在特定的时间段或者
超过某个特定的数量或任务类型发生变化时,系统会动态扩大线程池。
当线程池满足任务数量时,可以在系统的特定时期或者超过某些特定
的任务数量或任务类型发生变化时,动态缩小线程池,以减少系统的
负载。
线程池采用阻塞队列存储等待处理的任务,线程正常情况下从该
队列里获取任务进行执行,如果该队列中没有可执行的任务,线程处
于等待状态,只有任务加入队列,他才被启动继续工作。
线程池可以有效地提高系统的执行效率,降低 cpu 负载,它可
以动态扩大和缩小线程池,满足系统在各种不同状态下所需要的资源
数量,使系统性能受益于线程池的管理。
Java中深入分析线程池基本原理及实现机制
Java中深入分析线程池基本原理及实现机制(英文介绍):In Java, a thread pool is a crucial component for managing the execution of multiple threads, optimizing resource usage, and improving application performance. At its core, a thread pool works by maintaining a set of worker threads that are ready to execute tasks from a queue. This approach minimizes the overhead associated with thread creation and destruction, allowing the system to reuse threads efficiently.Basic Principles:1.Thread Reuse: Instead of creating a new thread for each task, a thread poolmaintains a pool of threads that can be reused. This significantly reduces the time and resources required for thread creation and teardown.2.Task Queuing: When a task is submitted to the thread pool, it is added to awork queue. Worker threads from the pool pick up tasks from this queue and execute them.3.Controlled Concurrency: By limiting the number of threads in the pool, thethread pool provides a mechanism to control concurrency, preventingexcessive resource usage and potential system overloads.4.Dynamic Scaling: Some thread pool implementations allow for dynamicscaling of the number of threads based on the workload. This ensuresoptimal resource utilization under varying demands.Implementation Mechanisms:Java provides several built-in thread pool implementations throughthe java.util.concurrent package, such as:1.FixedThreadPool: This pool maintains a fixed number of threads, regardlessof the workload. It is suitable for scenarios where the number of concurrenttasks is known and consistent.2.CachedThreadPool: This pool creates new threads as needed but alsoreclaims unused threads, keeping them available for later use. It is suitablefor scenarios with varying workloads.3.ScheduledThreadPool: This pool allows for the scheduling of tasks to run atfixed rates or with fixed delays. It is useful for periodic tasks or tasks thatrequire delayed execution.4.SingleThreadExecutor: This pool executes tasks in a single thread,providing a guarantee that tasks will be executed in the order they weresubmitted.Under the hood, these thread pools rely on a combination of data structures (such as blocking queues) and synchronization mechanisms (such as locks and condition variables) to coordinate task submission, execution, and completion. Java线程池基本原理及实现机制(中文介绍):在Java中,线程池是管理多个线程执行、优化资源使用和提高应用程序性能的关键组件。
深入分析java线程池的实现原理
深⼊分析java线程池的实现原理结合多篇⽂章整理,(仅⽤于⾯试)(相当于 copy,hhhh)前⾔线程是稀缺资源,如果被⽆限制的创建,不仅会消耗系统资源,还会降低系统的稳定性,合理的使⽤线程池对线程进⾏统⼀分配、调优和监控,有以下好处:1、降低资源消耗;2、提⾼响应速度;3、提⾼线程的可管理性。
Java1.5中引⼊的Executor框架把任务的提交和执⾏进⾏解耦,只需要定义好任务,然后提交给线程池,⽽不⽤关⼼该任务是如何执⾏、被哪个线程执⾏,以及什么时候执⾏。
demopublic class ExecutorTest {private static Executor executor = Executors.newFixedThreadPool(10);public static void main(String[] args) {for (int i = 0; i < 20; i++) {executor.execute(new Task());}}static class Task implements Runnable {@Overridepublic void run() {System.out.println(Thread.currentThread().getName());}}}1、Executors.newFixedThreadPool(10)初始化⼀个包含 10 个线程的线程池 executor ;2、通过executor.execute()⽅法提交 20 个任务,每个任务打印当前的线程名。
3、负责执⾏任务的线程的⽣命周期都由 Executor 框架进⾏管理;⼿写⼀个线程池为了更好的理解和分析关于线程池的源码,我们先来按照线程池的思想,⼿写⼀个⾮常简单的线程池。
其实很多时候⼀段功能代码的核⼼主逻辑可能没有多复杂,但是为了让核⼼流程顺利运⾏,就需要额外添加很多分⽀的辅助流程。
java自定义线程池的原理简介
java⾃定义线程池的原理简介线程池的相关概念就不在这⾥说明了,百度⼀下有很多,这⾥简单表述⼀下如何实现⼀个⾃定义的线程池就⾏线程管理,我们如果要实现⼀个线程池对线程的管理,那么需要实现⼀下⼏点的思路:1.如何管理线程2.如何定义⼯作线程以及⼯作线程如何持续的保持运⾏状态3.如何定义线程池⼤⼩及队列⼤⼩4.如何提供接⼝给调⽤者使⽤5.如何关闭线程池中的线程接下来我们就⼀⼀的实现这⼏个问题。
1.我们需要定义⼀个队列来来管理线程,这⾥使⽤了LinkedBlockingQueue// 1.定义⼀个存储线程队列private LinkedBlockingQueue<Runnable> queue;2.因为是⼀个简单的测试,所以我们可以先定义⼀个内部类来实现⼯作线程// 2.定义⼯作线程进⾏线程的执⾏class Worker extends Thread {private SelfThreadPoolExecutor threadPoolExecutor;public Worker(SelfThreadPoolExecutor poolExecutor) {this.threadPoolExecutor = poolExecutor;}@Overridepublic void run() {Runnable task;while (threadPoolExecutor.receiveTask || threadPoolExecutor.queue.size() > 0) {try {// 有线程则取出来,否则等待System.out.println("准备消费线程");task = threadPoolExecutor.queue.take();if (task != null) {task.run();System.out.println("消费线程");}} catch (InterruptedException e) {e.printStackTrace();}}}}SelfThreadPoolExecutor是外部定义的整体类名3.使⽤有参的构造⽅法进⾏线程池⼤⼩的管理// 3.存放⼯作线程的集合private List<Worker> workerList;// 4.线程池初始化public SelfThreadPoolExecutor(int coreSize, int queueSize) {if (coreSize <= 0 || queueSize <= 0) {throw new IllegalArgumentException("参数不正确");}this.queue = new LinkedBlockingQueue<>(queueSize);// 线程安全的集合this.workerList = Collections.synchronizedList(new ArrayList<>());for (int i = 0; i < coreSize; i++) {Worker worker = new Worker(this);worker.start();workerList.add(worker);}}4.定义阻塞和⾮阻塞的⽅式提供对应的接⼝// 5.⾮阻塞的⽅法接⼝public boolean offer(Runnable task) {if (receiveTask) {return queue.offer(task);} else {return false;}}// 6.阻塞的⽅法接⼝public void put(Runnable task) {try {if (receiveTask) {queue.put(task);}} catch (InterruptedException e) {e.printStackTrace();}}6.进⾏线程池的关闭// 7.线程池的关闭private boolean receiveTask = true;public void shutdown() {// 7.1.队列不再接收线程receiveTask = false;// 7.2.关闭处于wait或block的线程for (Thread thread : workerList) {if (Thread.State.BLOCKED.equals(thread.getState())|| Thread.State.WAITING.equals(thread.getState())|| Thread.State.TIMED_WAITING.equals(thread.getState())){thread.interrupt();}}}我们测试的⽅法如下:public static void main(String [] args){SelfThreadPoolExecutor selfThreadPoolExecutor = new SelfThreadPoolExecutor(5,10);for(int i = 0;i < 20;i++){Runnable task = () ->{System.out.println("开启线程");};selfThreadPoolExecutor.put(task);}selfThreadPoolExecutor.shutdown();}运⾏结果是:准备消费线程准备消费线程准备消费线程准备消费线程准备消费线程开启线程消费线程准备消费线程开启线程消费线程准备消费线程开启线程消费线程准备消费线程。
java线程池的大体原理
既然有了进程,什么缘故还要提出线程的概念呢?因为与创建一个新的进程相较,创建一个线程将会花费小得多的系统资源,关于一些小型的应用,可能感觉不到这点,但关于那些并发进程数专门多的应用,利用线程会比利用进程取得更好的性能,从而降低操作系统的负担。
另外,线程共享创建它的进程的全局变量,因此线程间的通信编程会更将简单,完全能够抛弃传统的进程间通信的IPC编程,而采纳共享全局变量来进行线程间通信。
有了上面那个概念,咱们下面就进入正题,来看一下线程池究竟是怎么一回事?其实线程池的原理很简单,类似于操作系统中的缓冲区的概念,它的流程如下:先启动假设干数量的线程,并让这些线程都处于睡眠状态,当客户端有一个新请求时,就会唤醒线程池中的某一个睡眠线程,让它来处置客户端的那个请求,当处置完那个请求后,线程又处于睡眠状态。
可能你或许会问:什么缘故要弄得这么麻烦,若是每当客户端有新的请求时,我就创建一个新的线程不就完了?这或许是个不错的方式,因为它能使得你编写代码相对容易一些,但你却忽略了一个重要的问题――性能!就拿我所在的单位来讲,我的单位是一个省级数据大集中的银行网络中心,顶峰期每秒的客户端请求并发数超过100,若是为每一个客户端请求创建一个新线程的话,那花费的CPU时刻和内存将是惊人的,若是采纳一个拥有200个线程的线程池,那将会节约大量的的系统资源,使得更多的CPU时刻和内存用来处置实际的商业应用,而不是频繁的线程创建与销毁。
既然一切都明白了,那咱们就开始着手实现一个真正的线程池吧,线程编程能够有多种语言来实现,例如C、C++、java等等,但不同的操作系统提供不同的线程API接口,为了让你能更明白线程池的原理而幸免陷入烦琐的API挪用当中,我采纳了JAVA语言来实现它,由于JAVA 语言是一种跨平台的语言,因此你没必要为利用不同的操作系统而无法编译运行本程序而苦恼,只要你安装了JDK1.2以上的版本,都能正确地编译运行本程序。
Java线程池详解(图解)
Java线程池详解(图解)来源:/p/098819be088c拓展: ⼿动创建 new ThreadPoolExecutor 的使⽤: https:///a/1190000011527245 https:///dweizhao/article/details/78603277 :销毁、⽴即停⽌前⾔Java中的线程池⼗分重要,⽆论是在实际应⽤中还是应对⾯试⼀、线程池原理1.1 使⽤线程池的好处第⼀:降低资源消耗。
通过重复利⽤已创建的线程降低线程创建和销毁造成的消耗。
第⼆:提⾼响应速度。
当任务到达时,任务可以不需要等到线程创建就能⽴即执⾏。
第三:提⾼线程的可管理性。
线程是稀缺资源,如果⽆限制地创建,不仅会消耗系统资源,还会降低系统的稳定性,使⽤线程池可以进⾏统⼀分配、调优和监控。
1.2 线程池的实现原理当向线程池提交任务后,线程池会按下图所⽰流程去处理这个任务1)线程池判断核⼼线程池⾥的线程是否都在执⾏任务。
如果不是,则创建⼀个新的⼯作线程来执⾏任务。
如果核⼼线程池⾥的线程都在执⾏任务,则进⼊下个流程。
2)线程池判断⼯作队列是否已经满。
如果⼯作队列没有满,则将新提交的任务存储在这个⼯作队列⾥。
如果⼯作队列满了,则进⼊下个流程。
3)线程池判断线程池的线程是否都处于⼯作状态。
如果没有,则创建⼀个新的⼯作线程来执⾏任务。
如果已经满了,则交给饱和策略来处理这个任务。
对应到代码层⾯就是ThreadPoolExecutor执⾏execute()⽅法。
如下图所⽰:1)如果当前运⾏的线程少于corePoolSize,则创建新线程来执⾏任务(注意,执⾏这⼀步骤需要获取全局锁)。
2)如果运⾏的线程等于或多于corePoolSize,则将任务加⼊BlockingQueue。
3)如果⽆法将任务加⼊BlockingQueue(队列已满),则创建新的线程来处理任务(注意,执⾏这⼀步骤需要获取全局锁)。
4)如果创建新线程将使当前运⾏的线程超出maximumPoolSize,任务将被拒绝,并调⽤RejectedExecutionHandler.rejectedExecution()⽅法。
线程池的原理及线程池的创建方式
线程池的原理及线程池的创建⽅式什么是线程池Java中的线程池是运⽤场景最多的并发框架,⼏乎所有需要异步或并发执⾏任务的程序都可以使⽤线程池。
在开发过程中,合理地使⽤线程池能够带来3个好处。
第⼀:降低资源消耗。
通过重复利⽤机制已降低线程创建和销毁造成的消耗。
第⼆:提⾼响应速度。
当任务到达时,任务可以不需要等到线程创建就能⽴即执⾏。
第三:提⾼线程的可管理性。
线程是稀缺资源,如果⽆限制地创建,不仅会消耗系统资源,还会降低系统的稳定性,使⽤线程池可以进⾏统⼀分配、调优和监控。
但是,要做到合理利⽤线程池,必须对其实现原理了如指掌。
线程池作⽤线程池是为突然⼤量爆发的线程设计的,通过有限的⼏个固定线程为⼤量的操作服务,减少了创建和销毁线程所需的时间,从⽽提⾼效率。
如果⼀个线程的时间⾮常长,就没必要⽤线程池了(不是不能作长时间操作,⽽是不宜。
),况且我们还不能控制线程池中线程的开始、挂起、和中⽌。
线程池原理剖析1.如果当前线程池中的线程数⽬⼩于corePoolSize,则每来⼀个任务,就会创建⼀个线程去执⾏这个任务;2.如果当前线程池中的线程数⽬>=corePoolSize,则每来⼀个任务,会尝试将其添加到任务缓存队列当中,若添加成功,则该任务会等待空闲线程将其取出去执⾏;若添加失败(⼀般来说是任务缓存队列已满),则会尝试创建新的线程去执⾏这个任务;3.如果队列已经满了,则在总线程数不⼤于maximumPoolSize的前提下,则创建新的线程4.如果当前线程池中的线程数⽬达到maximumPoolSize,则会采取任务拒绝策略进⾏处理;5.如果线程池中的线程数量⼤于 corePoolSize时,如果某线程空闲时间超过keepAliveTime,线程将被终⽌,直⾄线程池中的线程数⽬不⼤于corePoolSize;如果允许为核⼼池中的线程设置存活时间,那么核⼼池中的线程空闲时间超过keepAliveTime,线程也会被终⽌。
- 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
- 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
- 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。
在什么情况下使用线程池?1.单个任务处理的时间比较短2.将需处理的任务的数量大使用线程池的好处:1.减少在创建和销毁线程上所花的时间以及系统资源的开销2.如不使用线程池,有可能造成系统创建大量线程而导致消耗完系统内存以及”过度切换”。
线程池工作原理:线程池为线程生命周期开销问题和资源不足问题提供了解决方案。
通过对多个任务重用线程,线程创建的开销被分摊到了多个任务上。
其好处是,因为在请求到达时线程已经存在,所以无意中也消除了线程创建所带来的延迟。
这样,就可以立即为请求服务,使应用程序响应更快。
而且,通过适当地调整线程池中的线程数目,也就是当请求的数目超过某个阈值时,就强制其它任何新到的请求一直等待,直到获得一个线程来处理为止,从而可以防止资源不足。
线程池的替代方案线程池远不是服务器应用程序内使用多线程的唯一方法。
如同上面所提到的,有时,为每个新任务生成一个新线程是十分明智的。
然而,如果任务创建过于频繁而任务的平均处理时间过短,那么为每个任务生成一个新线程将会导致性能问题。
另一个常见的线程模型是为某一类型的任务分配一个后台线程与任务队列。
AWT 和 Swing 就使用这个模型,在这个模型中有一个 GUI 事件线程,导致用户界面发生变化的所有工作都必须在该线程中执行。
然而,由于只有一个 AWT 线程,因此要在 AWT 线程中执行任务可能要花费相当长时间才能完成,这是不可取的。
因此,Swing 应用程序经常需要额外的工作线程,用于运行时间很长的、同 UI 有关的任务。
每个任务对应一个线程方法和单个后台线程(single-background-thread)方法在某些情形下都工作得非常理想。
每个任务一个线程方法在只有少量运行时间很长的任务时工作得十分好。
而只要调度可预见性不是很重要,则单个后台线程方法就工作得十分好,如低优先级后台任务就是这种情况。
然而,大多数服务器应用程序都是面向处理大量的短期任务或子任务,因此往往希望具有一种能够以低开销有效地处理这些任务的机制以及一些资源管理和定时可预见性的措施。
线程池提供了这些优点。
工作队列就线程池的实际实现方式而言,术语“线程池”有些使人误解,因为线程池“明显的”实现在大多数情形下并不一定产生我们希望的结果。
术语“线程池”先于Java 平台出现,因此它可能是较少面向对象方法的产物。
然而,该术语仍继续广泛应用着。
虽然我们可以轻易地实现一个线程池类,其中客户机类等待一个可用线程、将任务传递给该线程以便执行、然后在任务完成时将线程归还给池,但这种方法却存在几个潜在的负面影响。
例如在池为空时,会发生什么呢?试图向池线程传递任务的调用者都会发现池为空,在调用者等待一个可用的池线程时,它的线程将阻塞。
我们之所以要使用后台线程的原因之一常常是为了防止正在提交的线程被阻塞。
完全堵住调用者,如在线程池的“明显的”实现的情况,可以杜绝我们试图解决的问题的发生。
我们通常想要的是同一组固定的工作线程相结合的工作队列,它使用 wait() 和// If we don't catch RuntimeException,// the pool could leak threadstry {r.run();}catch (RuntimeException e) {// You might want to log something here}}}}}您可能已经注意到了清单 1 中的实现使用的是 notify() 而不是notifyAll() 。
大多数专家建议使用 notifyAll() 而不是 notify() ,而且理由很充分:使用 notify() 具有难以捉摸的风险,只有在某些特定条件下使用该方法才是合适的。
另一方面,如果使用得当, notify() 具有比 notifyAll() 更可取的性能特征;特别是, notify() 引起的环境切换要少得多,这一点在服务器应用程序中是很重要的。
清单 1 中的示例工作队列满足了安全使用 notify() 的需求。
因此,请继续,在您的程序中使用它,但在其它情形下使用 notify() 时请格外小心。
使用线程池的风险虽然线程池是构建多线程应用程序的强大机制,但使用它并不是没有风险的。
用线程池构建的应用程序容易遭受任何其它多线程应用程序容易遭受的所有并发风险,诸如同步错误和死锁,它还容易遭受特定于线程池的少数其它风险,诸如与池有关的死锁、资源不足和线程泄漏。
死锁任何多线程应用程序都有死锁风险。
当一组进程或线程中的每一个都在等待一个只有该组中另一个进程才能引起的事件时,我们就说这组进程或线程死锁了。
死锁的最简单情形是:线程 A 持有对象 X 的独占锁,并且在等待对象 Y 的锁,而线程 B 持有对象 Y 的独占锁,却在等待对象 X 的锁。
除非有某种方法来打破对锁的等待(Java 锁定不支持这种方法),否则死锁的线程将永远等下去。
虽然任何多线程程序中都有死锁的风险,但线程池却引入了另一种死锁可能,在那种情况下,所有池线程都在执行已阻塞的等待队列中另一任务的执行结果的任务,但这一任务却因为没有未被占用的线程而不能运行。
当线程池被用来实现涉及许多交互对象的模拟,被模拟的对象可以相互发送查询,这些查询接下来作为排队的任务执行,查询对象又同步等待着响应时,会发生这种情况。
资源不足线程池的一个优点在于:相对于其它替代调度机制(有些我们已经讨论过)而言,它们通常执行得很好。
但只有恰当地调整了线程池大小时才是这样的。
线程消耗包括内存和其它系统资源在内的大量资源。
除了 Thread 对象所需的内存之外,每个线程都需要两个可能很大的执行调用堆栈。
除此以外,JVM 可能会为每个Java 线程创建一个本机线程,这些本机线程将消耗额外的系统资源。
最后,虽然线程之间切换的调度开销很小,但如果有很多线程,环境切换也可能严重地影响程序的性能。
如果线程池太大,那么被那些线程消耗的资源可能严重地影响系统性能。
在线程之间进行切换将会浪费时间,而且使用超出比您实际需要的线程可能会引起资源匮乏问题,因为池线程正在消耗一些资源,而这些资源可能会被其它任务更有效地利用。
除了线程自身所使用的资源以外,服务请求时所做的工作可能需要其它资源,例如 JDBC 连接、套接字或文件。
这些也都是有限资源,有太多的并发请求也可能引起失效,例如不能分配 JDBC 连接。
并发错误线程池和其它排队机制依靠使用 wait() 和 notify() 方法,这两个方法都难于使用。
如果编码不正确,那么可能丢失通知,导致线程保持空闲状态,尽管队列中有工作要处理。
使用这些方法时,必须格外小心;即便是专家也可能在它们上面出错。
而最好使用现有的、已经知道能工作的实现,例如在下面的无须编写您自己的池中讨论的 util.concurrent 包。
线程泄漏各种类型的线程池中一个严重的风险是线程泄漏,当从池中除去一个线程以执行一项任务,而在任务完成后该线程却没有返回池时,会发生这种情况。
发生线程泄漏的一种情形出现在任务抛出一个 RuntimeException 或一个 Error 时。
如果池类没有捕捉到它们,那么线程只会退出而线程池的大小将会永久减少一个。
当这种情况发生的次数足够多时,线程池最终就为空,而且系统将停止,因为没有可用的线程来处理任务。
有些任务可能会永远等待某些资源或来自用户的输入,而这些资源又不能保证变得可用,用户可能也已经回家了,诸如此类的任务会永久停止,而这些停止的任务也会引起和线程泄漏同样的问题。
如果某个线程被这样一个任务永久地消耗着,那么它实际上就被从池除去了。
对于这样的任务,应该要么只给予它们自己的线程,要么只让它们等待有限的时间。
请求过载仅仅是请求就压垮了服务器,这种情况是可能的。
在这种情形下,我们可能不想将每个到来的请求都排队到我们的工作队列,因为排在队列中等待执行的任务可能会消耗太多的系统资源并引起资源缺乏。
在这种情形下决定如何做取决于您自己;在某些情况下,您可以简单地抛弃请求,依靠更高级别的协议稍后重试请求,您也可以用一个指出服务器暂时很忙的响应来拒绝请求。
有效使用线程池的准则只要您遵循几条简单的准则,线程池可以成为构建服务器应用程序的极其有效的方法:∙不要对那些同步等待其它任务结果的任务排队。
这可能会导致上面所描述的那种形式的死锁,在那种死锁中,所有线程都被一些任务所占用,这些任务依次等待排队任务的结果,而这些任务又无法执行,因为所有的线程都很忙。
∙在为时间可能很长的操作使用合用的线程时要小心。
如果程序必须等待诸如 I/O 完成这样的某个资源,那么请指定最长的等待时间,以及随后是失效还是将任务重新排队以便稍后执行。
这样做保证了:通过将某个线程释放给某个可能成功完成的任务,从而将最终取得某些进展。
∙理解任务。
要有效地调整线程池大小,您需要理解正在排队的任务以及它们正在做什么。
它们是 CPU 限制的(CPU-bound)吗?它们是 I/O 限制的(I/O-bound)吗?您的答案将影响您如何调整应用程序。
如果您有不同的任务类,这些类有着截然不同的特征,那么为不同任务类设置多个工作队列可能会有意义,这样可以相应地调整每个池。
调整池的大小调整线程池的大小基本上就是避免两类错误:线程太少或线程太多。
幸运的是,对于大多数应用程序来说,太多和太少之间的余地相当宽。
请回忆:在应用程序中使用线程有两个主要优点,尽管在等待诸如 I/O 的慢操作,但允许继续进行处理,并且可以利用多处理器。
在运行于具有 N 个处理器机器上的计算限制的应用程序中,在线程数目接近 N 时添加额外的线程可能会改善总处理能力,而在线程数目超过 N 时添加额外的线程将不起作用。
事实上,太多的线程甚至会降低性能,因为它会导致额外的环境切换开销。
线程池的最佳大小取决于可用处理器的数目以及工作队列中的任务的性质。
若在一个具有 N 个处理器的系统上只有一个工作队列,其中全部是计算性质的任务,在线程池具有 N 或 N+1 个线程时一般会获得最大的 CPU 利用率。
对于那些可能需要等待 I/O 完成的任务(例如,从套接字读取 HTTP 请求的任务),需要让池的大小超过可用处理器的数目,因为并不是所有线程都一直在工作。
通过使用概要分析,您可以估计某个典型请求的等待时间(WT)与服务时间(ST)之间的比例。
如果我们将这一比例称之为 WT/ST,那么对于一个具有 N 个处理器的系统,需要设置大约 N*(1+WT/ST) 个线程来保持处理器得到充分利用。
处理器利用率不是调整线程池大小过程中的唯一考虑事项。
随着线程池的增长,您可能会碰到调度程序、可用内存方面的限制,或者其它系统资源方面的限制,该文章里有个例子,简单的描述了线程池的内部实现,建议根据里面的例子来了解JAVA 线程池的原理。