给线程池增加自动扩充线程数量,以及闲时自动回收的功能

package com.dwz.concurrency.chapter13;

import java.util.ArrayList;
import java.util.Iterator;
import java.util.LinkedList;
import java.util.List; public class SimpleThreadPool4 extends Thread {
private static int size;
private final int queueSize;
private final static int DEFAULT_TASK_QUEUE_SIZE = 2000;
private static volatile int seq = 0;
private final static String THREAD_PREFIX = "SIMPLE_THREAD_POOL-";
private final static ThreadGroup GROUP = new ThreadGroup("Pool_Group");
private final static LinkedList<Runnable> TASK_QUEUE = new LinkedList<>();
private final static List<WorkerTask> THREAD_QUEUE = new ArrayList<>();
private final DiscardPolicy discardPolicy;
private final static DiscardPolicy DEFAULT_DISCARD_POLICY = () -> {
throw new DiscardException("Discard this task.");
};
private volatile boolean destroy = false;
private int min;
private int max;
private int active; public SimpleThreadPool4() {
this(4, 8, 12, DEFAULT_TASK_QUEUE_SIZE, DEFAULT_DISCARD_POLICY);
} public SimpleThreadPool4(int min, int active, int max, int queueSize, DiscardPolicy discardPolicy) {
this.min = min;
this.active = active;
this.max = max;
this.queueSize = queueSize;
this.discardPolicy = discardPolicy;
init();
} private void init() {
for (int i = 0; i < this.min; i++) {
createWorkTask();
}
this.size = min;
this.start();
} public void submit(Runnable runnable) {
if (destroy) {
throw new IllegalStateException("The thread pool already destroy and not allow submit task.");
}
synchronized (TASK_QUEUE) {
if (TASK_QUEUE.size() >= this.queueSize) {
discardPolicy.discard();
}
TASK_QUEUE.addLast(runnable);
TASK_QUEUE.notifyAll();
}
} @Override
public void run() {
while (!destroy) {
System.out.printf("Pool#Min:%d,Active:%d,Max:%d,Current:%d,QueueSize:%d,ThreadQueueSize::%d\n",
this.min, this.active, this.max, this.size, TASK_QUEUE.size(), THREAD_QUEUE.size());
try {
Thread.sleep(5_000L);
if(TASK_QUEUE.size() > this.active && this.size < this.active) {
for(int i = this.size; i < this.active; i++) {
createWorkTask();
}
System.err.printf("The pool incremented to active. currentSize:%d\n", this.size);
this.size = this.active;
} else if(TASK_QUEUE.size() > this.max && this.size < this.max) {
for(int i = this.size; i < this.max; i++) {
createWorkTask();
}
System.err.printf("The pool incremented to max currentSize:%d\n", this.size);
this.size = this.max;
} if(TASK_QUEUE.isEmpty() && this.size > this.active) {
System.out.println("===============Reduce===============");
synchronized (THREAD_QUEUE) {
int releaseSize = this.size - this.active;
for(Iterator<WorkerTask> it = THREAD_QUEUE.iterator(); it.hasNext();) {
if(releaseSize <= 0) {
break;
}
WorkerTask task = it.next();
task.close();
task.interrupt();
it.remove();
releaseSize--;
}
this.size = this.active;
}
}
} catch (InterruptedException e) {
e.printStackTrace();
}
}
} private void createWorkTask() {
WorkerTask task = new WorkerTask(GROUP, THREAD_PREFIX + (seq++));
task.start();
THREAD_QUEUE.add(task);
} public void shutdown() throws InterruptedException {
while (!TASK_QUEUE.isEmpty()) {
Thread.sleep(50);
}
synchronized (THREAD_QUEUE) {
int initVal = THREAD_QUEUE.size();
while (initVal > 0) {
for (WorkerTask task : THREAD_QUEUE) {
if (task.getTaskState() == TaskState.BLOCKED) {
task.interrupt();
task.close();
initVal--;
} else {
Thread.sleep(10);
}
}
}
}
this.destroy = true;
System.out.println("The thread pool disposed.");
System.err.println("THREAD_QUEUE size is: " + THREAD_QUEUE.size());
System.out.println("GROUP.activeCount() is: " + GROUP.activeCount());
} public int getSize() {
return size;
} public int getQueueSize() {
return queueSize;
} public int getMin() {
return min;
} public int getMax() {
return max;
} public int getActive() {
return active;
} public boolean isDestroy() {
return destroy;
} private enum TaskState {
FREE, RUNNING, BLOCKED, DEAD
} public static class DiscardException extends RuntimeException { public DiscardException(String message) {
super(message);
}
} public interface DiscardPolicy {
void discard() throws DiscardException;
} private static class WorkerTask extends Thread {
private volatile TaskState taskState = TaskState.FREE; public WorkerTask(ThreadGroup group, String name) {
super(group, name);
} public TaskState getTaskState() {
return this.taskState;
} @Override
public void run() {
OUTER: while (this.taskState != TaskState.DEAD) {
Runnable runnable = null;
synchronized (TASK_QUEUE) {
while (TASK_QUEUE.isEmpty()) {
try {
this.taskState = TaskState.BLOCKED;
TASK_QUEUE.wait();
} catch (InterruptedException e) {
e.printStackTrace();
System.err.println("closed.");
// 线程被打断回到OUTER位置
break OUTER;
}
}
runnable = TASK_QUEUE.removeFirst();
} if (runnable != null) {
System.out.println("runnable into...");
this.taskState = TaskState.RUNNING;
runnable.run();
this.taskState = TaskState.FREE;
}
}
} public void close() {
this.taskState = TaskState.DEAD;
}
} public static void main(String[] args) throws InterruptedException {
SimpleThreadPool4 threadPool = new SimpleThreadPool4();
for (int i = 0; i < 100; i++) {
threadPool.submit(() -> {
System.out.println("The runnable be serviced by " + Thread.currentThread() + " start.");
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("The runnable be serviced by " + Thread.currentThread() + " finished.");
});
}
}
}

SimpleThreadPool给线程池增加自动扩充线程数量,以及闲时自动回收的功能的更多相关文章

  1. SimpleThreadPool给线程池增加拒绝策略和停止方法

    给线程池增加拒绝策略和停止方法 package com.dwz.concurrency.chapter13; import java.util.ArrayList; import java.util. ...

  2. 线程池如何复用一个线程-- ThreadPoolExecutor的实现(未完)

    任务是一组逻辑工作单元,而线程则是使任务异步执行的机制.在Java中,Runnable对象代表一个任务,Thread对象负责创建一个线程执行这个任务. 前提:1. 程序需要处理大量任务 2. 任务的执 ...

  3. 使用Callable接口创建线程和使用线程池的方式创建线程

    1.使用Callable接口的方式实现多线程,这是JDK5.0新增的一种创建多线程的方法 package com.baozi.java2; import java.util.concurrent.Ca ...

  4. 发一个可伸缩线程池大小的python线程池。已通过测试。

    发一个可伸缩线程池大小的线程池. 当任务不多时候,不开那么多线程,当任务多的时候开更多线程.当长时间没任务时候,将线程数量减小到一定数量. java的Threadpoolexcutor可以这样,py的 ...

  5. 线程池系列一:线程池作用及Executors方法讲解

    线程池的作用: 线程池作用就是限制系统中执行线程的数量.     根据系统的环境情况,可以自动或手动设置线程数量,达到运行的最佳效果:少了浪费了系统资源,多了造成系统拥挤效率不高.用线程池控制线程数量 ...

  6. 线程池;java的线程池的实现原理;适用于频繁互动(如电商网站)

    线程池是一种多线程处理形式,处理过程中将任务加入到队列,然后在创建线程后自己主动启动这些任务.线程池线程都是后台线程.每一个线程都使用默认的堆栈大小,以默认的优先级执行.并处于多线程单元中. 假设某个 ...

  7. java多线程系类:JUC线程池:05之线程池原理(四)(转)

    概要 本章介绍线程池的拒绝策略.内容包括:拒绝策略介绍拒绝策略对比和示例 转载请注明出处:http://www.cnblogs.com/skywang12345/p/3512947.html 拒绝策略 ...

  8. java多线程系类:JUC线程池:03之线程池原理(二)(转)

    概要 在前面一章"Java多线程系列--"JUC线程池"02之 线程池原理(一)"中介绍了线程池的数据结构,本章会通过分析线程池的源码,对线程池进行说明.内容包 ...

  9. java多线程系类:JUC线程池:04之线程池原理(三)(转)

    转载请注明出处:http://www.cnblogs.com/skywang12345/p/3509960.html 本章介绍线程池的生命周期.在"Java多线程系列--"基础篇& ...

随机推荐

  1. SYN攻击源码

    一.linux下源代码实现/* syn flood by wqfhenanxc. * random soruce ip and random sourec port. * use #include & ...

  2. linux_文本编译使用命令

    一:字符模式与shell命令 字符界面和图形界面 字符界面优点: 1):系统执行效率高,稳定性高,执行结果可直接返回 2):节省系统资源,对一个服务器至关重要 3):节省大量网络开销,大幅降低运行成本 ...

  3. Filebeat7 Kafka Gunicorn Flask Web应用程序日志采集

    本文的内容 如何用filebeat kafka es做一个好用,好管理的日志收集工具 放弃logstash,使用elastic pipeline gunicron日志格式与filebeat/es配置 ...

  4. Tomcat 设置80端口

    1:修改tomcat配置 vi /usr/local/tomcat/conf/server.xml 找到 Connector port="8080" protocol=" ...

  5. O009、KVM 网络虚拟化基础

    参考https://www.cnblogs.com/CloudMan6/p/5289590.html   网络虚拟化是虚拟化技术中最复杂的部分,学习难度最大.   但因为网络是虚拟化中非常重要的资源, ...

  6. debezium关于cdc的使用(上)

    博文原址:debezium关于cdc的使用(上) 简介 debezium是一个为了捕获数据变更(cdc)的开源的分布式平台.启动并指向数据库,当其他应用对此数据库执行inserts.updates.d ...

  7. 让IE6、IE7、IE8、IE9、IE10、IE11支持Bootstrap的解决方法

    最近做一个Web网站,之前一直觉得bootstrap非常好,这次使用了bootstrap3,在chrome,firefox,safari,opera,360浏览器(极速模式).搜狗浏览器等浏览器下均没 ...

  8. 【转载】关于java 的InputStream和OutputStream的理解

    关于InputStream和OutputStream的输入输出方向的理解 InputStream输入类,首先需要读取的内容转化成输入流,再从它那里进行读取,先关联源:之后过程中关联目的,这样形成了流: ...

  9. Spring Boot整合拦截器

    过滤器和监听器都属于Servlet 的api,还可以使用 Spring 提供的拦截器(HandlerInterceptor)进行改更精细的控制.

  10. HDU - 6704 K-th occurrence (后缀数组+主席树/后缀自动机+线段树合并+倍增)

    题意:给你一个长度为n的字符串和m组询问,每组询问给出l,r,k,求s[l,r]的第k次出现的左端点. 解法一: 求出后缀数组,按照排名建主席树,对于每组询问二分或倍增找出主席树上所对应的的左右端点, ...