发布日期:2025-05-10 16:00 点击次数:199
[[418595]]
为擢升科罚才略和并发度,Web容器一般会把科罚苦求的任务放到线程池,而JDK的原生线程池先天符合CPU密集型任务,并不符合咱们频繁的 I/O 密集任务科罚,于是Tomcat纠正之。
Tomcat 线程池旨趣其实ThreadPoolExecutor的参数主要有如下要津点:
闭幕线程个数而Tomcat对这俩资源皆需要闭幕,不然高并发下CPU、内存皆有被浮滥可能。因此Tomcat的线程池传参:
// 定制的任务部队 taskqueue = new TaskQueue(maxQueueSize); // 定制的线程工场 TaskThreadFactory tf = new TaskThreadFactory(namePrefix, daemon, getThreadPriority() ); // 定制线程池 executor = new ThreadPoolExecutor(getMinSpareThreads(), getMaxThreads(), maxIdleTime, TimeUnit.MILLISECONDS, taskqueue, tf);
Tomcat对线程数也有闭幕,设立:
中枢线程数(minSpareThreads) 最大线程池数(maxThreads)Tomcat线程池还有我方的特点任务科罚经由,通过重写execute步调结束了我方的特点任务科罚逻辑:
前corePoolSize个任务时,来一个任务就创建一个新线程 再有任务,就把任务放入任务部队,让通盘线程去抢。若部队满,就创建临时线程 总线程数达到maximumPoolSize,则延续尝试把任务放入任务部队 若缓冲部队也满了,插入失败,践诺拒绝政策和 JDK 线程池的分袂就在step3,Tomcat在线程总额达到最大数时,不是立即践诺拒绝政策,而是再尝试向任务部队添加任务,添加失败后再践诺拒绝政策。
迷水商城具体又是怎么结束的呢?
迷水商城public void execute(Runnable command, long timeout, TimeUnit unit) { submittedCount.incrementAndGet(); try { // 调用JDK原生线程池的execute践诺任务 super.execute(command); } catch (RejectedExecutionException rx) { // 总线程数达到maximumPoolSize后,JDK原生线程池会践诺默许拒绝政策 if (super.getQueue() instanceof TaskQueue) { final TaskQueue queue = (TaskQueue)super.getQueue(); try { // 延续尝试把任务放入任务部队 if (!queue.force(command, timeout, unit)) { submittedCount.decrementAndGet(); // 若缓冲部队如故满了,插入失败,践诺拒绝政策。 throw new RejectedExecutionException("..."); } } } } }定制任务部队
Tomcat线程池的execute步调第一溜:
迷水商城迷水商城submittedCount.incrementAndGet();
任求践诺失败,迷药拼多多抛相称时,将该计数器减一:
迷水商城迷水商城submittedCount.decrementAndGet();
Tomcat线程池使用 submittedCount 变量调遣已提交到线程池,但未践诺完的任务数目。
迷水商城为何要调遣这么一个变量呢?
迷水商城Tomcat的任务部队TaskQueue推广了JDK的LinkedBlockingQueue,Tomcat给了它一个capacity,传给父类LinkedBlockingQueue的构造器。
迷水商城public class TaskQueue extends LinkedBlockingQueue<Runnable> { public TaskQueue(int capacity) { super(capacity); } ... }
capacity参数通过Tomcat的 maxQueueSize 参数设立,但maxQueueSize默许值为Integer.MAX_VALUE:这么,面前方程数达到中枢线程数后,再来的任务,线程池会把任务添加到任务部队,何况总会到手,就耐久无契机创建新线程了。
为此,TaskQueue重写了LinkedBlockingQueue#offer,在合应时机复返false,暗示任务添加失败,线程池此时会创建新的线程。
迷水商城迷水商城
什么叫合应时机?
public class TaskQueue extends LinkedBlockingQueue<Runnable> { ... @Override // 线程池调用任务部队的步调时,面前方程数 > core线程数 public boolean offer(Runnable o) { // 若线程数已达max,则弗成创建新线程,只可放入任务部队 if (parent.getPoolSize() == parent.getMaximumPoolSize()) return super.offer(o); // 至此,标明 max线程数 > 面前方程数 > core线程数 // 诠释可创建新线程: // 1. 若已提交任务数 < 面前方程数 // 标明还有清闲线程,无需创建新线程 if (parent.getSubmittedCount()<=(parent.getPoolSize())) return super.offer(o); // 2. 若已提交任务数 > 面前方程数 // 线程不够用了,复返false去创建新线程 if (parent.getPoolSize()<parent.getMaximumPoolSize()) return false; // 默许情况下老是把任务放入任务部队 return super.offer(o); } }
是以Tomcat调遣 已提交任务数 是为了在职务部队长度无穷时,让线程池还能有契机创建新线程。
上一篇:央视315晚会曝光“保水虾仁”后,湛江4家涉事企业被罚超800万元
下一篇:“天下玉,扬州工”,扬州古玉到颐和园了!