浅析java中的四种线程池
1.使用线程池的好处

2.JUC中几种常用的线程池
java.util.concurrent包下的Executors工厂类,提供了一系列的线程池的创建方法,其构造方法如下:
public ThreadPoolExecutor(int corePoolSize, //线程池线程核心数量,线程池维护线程的最少数量
int maximumPoolSize, //线程池最大线程数量
long keepAliveTime, //空闲线程存活时间
TimeUnit unit, //存活时间的时间单位
BlockingQueue<Runnable> workQueue, //存储任务的线程队列
ThreadFactory threadFactory, //线程工厂
RejectedExecutionHandler handler) { //拒绝策略
if (corePoolSize < 0 ||
maximumPoolSize <= 0 ||
maximumPoolSize < corePoolSize ||
keepAliveTime < 0)
throw new IllegalArgumentException();
if (workQueue == null || threadFactory == null || handler == null)
throw new NullPointerException();
this.corePoolSize = corePoolSize;
this.maximumPoolSize = maximumPoolSize;
this.workQueue = workQueue;
this.keepAliveTime = unit.toNanos(keepAliveTime);
this.threadFactory = threadFactory;
this.handler = handler;
}
其中常用的线程池有四种,分别是fixedThreadPool、cachedThreadPool、ScheduledThreadPool和SingleThreadExecutor。他们分别适用在不同的场合。
2.1 newFixedThreadPool
特点:
- 用于创建一个可重用、固定线程数量的线程池;
- 当线程池中线程都处于运行中时,新来的线程会进入等待状态,直到线程池中出现一个空闲线程;
- 当一个线程在任务中途退出、终止时,会有一个新的线程来替代它继续完成后面未完成的任务。
- 除非采用显式关闭的方法去关闭某个线程,否则线程会一直存在,不会释放资源。
- 任务存储在无界阻塞队列中
- 适用场景:长期任务
构造方法:
public static ExecutorService newFixedThreadPool(int nThreads) {
return new ThreadPoolExecutor(nThreads, nThreads,
0L, TimeUnit.MILLISECONDS,
new LinkedBlockingQueue<Runnable>());
}
实例代码:
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import org.junit.Test; public class ExecutorsTest extends Thread{
private int index;
public ExecutorsTest(int i)
{
this.index = i;
} public void run()
{
try
{
System.out.println("[Thread"+this.index+"]" +"start..");
Thread.sleep((int)(Math.random()*10000));
System.out.println("[Thread"+this.index+"]" + "end");
}catch(Exception e)
{
e.printStackTrace();
}
} public static void main(String args[])
{
ExecutorService service = Executors.newFixedThreadPool(4); for(int i=0;i<10;i++)
{
service.execute(new ExecutorsTest(i));
}
service.shutdown(); }
}

因为线程池中线程数量一共有4个,所以当一次有大于4个的任务需要执行时,因为线程池中无空闲线程,后续任务进入等待状态,当其他任务执行完毕后,线程空闲,则马上开始执行正在等待的任务。
2.1 newCachedThreadPool
特点:
- 线程池数量上限为:Integer.MaxValue(2147483647);
- 线程池默认空闲60S,超过60S会从线程池中移除;
- 新来任务时,先检查是否有空闲线程可使用,若无,则创建一个新线程执行任务;
- 任务存储在同步队列中。
- 适用场景:短时异步任务。
构造函数:
public static ExecutorService newCachedThreadPool() {
return new ThreadPoolExecutor(0, Integer.MAX_VALUE,
60L, TimeUnit.SECONDS,
new SynchronousQueue<Runnable>());
}
示例代码:
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors; import org.junit.Test; public class ExecutorsTest extends Thread{
private int index;
public ExecutorsTest(int i)
{
this.index = i;
} public void run()
{
try
{
System.out.println("[Thread"+this.index+"]" +"start..");
Thread.sleep((int)(Math.random()*1000));
System.out.println("[Thread"+this.index+"]" + "end");
}catch(Exception e)
{
e.printStackTrace();
}
} public static void main(String args[])
{
ExecutorService service = Executors.newCachedThreadPool(); for(int i=0;i<10;i++)
{
service.execute(new ExecutorsTest(i));
}
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} for(int i=0;i<5;i++)
{
service.execute(new ExecutorsTest(i));
} service.shutdown(); }

创建10个工作线程用于处理任务,当线程执行完毕后,处于空闲状态,此时若出现新的任务,则会从线程池中用空闲的线程来处理新的任务。若没有空闲线程,则开启新线程处理。
2.3 newSingleThreadExecutor
特点:
- 创建一个单个Worker的线程;
- 线程会按照顺序依次执行;
- 任务存储在无界阻塞队列中
- 适用场景:需要按照顺序执行的任务。
构造方法:
public static ExecutorService newSingleThreadExecutor() {
return new FinalizableDelegatedExecutorService
(new ThreadPoolExecutor(1, 1,
0L, TimeUnit.MILLISECONDS,
new LinkedBlockingQueue<Runnable>()));
}
实例代码:
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors; import org.junit.Test; public class ExecutorsTest extends Thread{
private int index;
public ExecutorsTest(int i)
{
this.index = i;
} public void run()
{
try
{
System.out.println("[Thread"+this.index+"]" +"start..");
Thread.sleep((int)(Math.random()*1000));
System.out.println("[Thread"+this.index+"]" + "end");
}catch(Exception e)
{
e.printStackTrace();
}
} public static void main(String args[])
{
ExecutorService service = Executors.newSingleThreadExecutor(); for(int i=0;i<10;i++)
{
service.execute(new ExecutorsTest(i));
}
service.shutdown(); } }

出现多个任务时,SingleThreadExecutor会按照顺序依次执行各个任务。
2.4 newScheduledThreadPool
特点:
- 任务存储在无界延迟队列中
- 适用场景:需要定期执行或延迟执行的任务
构造方法:
public static ScheduledExecutorService newScheduledThreadPool(int corePoolSize) {
return new ScheduledThreadPoolExecutor(corePoolSize);
}
public ScheduledThreadPoolExecutor(int corePoolSize) {
super(corePoolSize, Integer.MAX_VALUE, 0, NANOSECONDS,
new DelayedWorkQueue());
}
实例代码一(scheduleAtFixedRate的使用):
import java.util.Date;
import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.TimeUnit; public class ExecutorsTest2 extends Thread{
private int index;
public ExecutorsTest2()
{
} public void run()
{
try
{
System.out.println("[Current Time is "+new Date().toString()); }catch(Exception e)
{
e.printStackTrace();
}
} public static void main(String args[])
{
/*
* 执行定时任务newScheduledThreadPool
*/
ScheduledExecutorService service = Executors.newScheduledThreadPool(10); //5秒后开始执行,每隔一秒执行一次
service.scheduleAtFixedRate(new ExecutorsTest2(), 5, 1, TimeUnit.SECONDS);
}
}

scheduleAtFixedRate方法,一共四个参数,分别是:需要执行的任务task、延迟执行时间t1、每次执行任务的时间间隔t2、时间间隔单位。
含义是:在t1时间过后,以 1次/t2 的频率来不断执行 task。
上述代码中,在5秒延迟后,以 1次/1秒的频率执行 打印当前时间的任务。
实例代码二(scheduleWithFixedDelay的使用):
import java.util.Date;
import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.TimeUnit; public class ExecutorsTest3 extends Thread{
private int index;
public ExecutorsTest3()
{
} public void run()
{
try
{
//每次任务大约耗时1秒
Thread.sleep(1000);
System.out.println("[Current Time is "+new Date().toString()); }catch(Exception e)
{
e.printStackTrace();
}
} public static void main(String args[])
{
/*
* 执行定时任务newScheduledThreadPool
*/
ScheduledExecutorService service = Executors.newScheduledThreadPool(10); //5秒后开始执行,每次任务执行完后延迟3秒后,继续执行下一次
service.scheduleWithFixedDelay(new ExecutorsTest3(), 5, 3, TimeUnit.SECONDS);
}
}

scheduleWithFixedDelay也是四个参数,分别是:待执行的任务Task,延迟时间t1,每次任务执行完毕后延迟t2秒后执行下次任务,延迟时间单位。
浅析java中的四种线程池的更多相关文章
- Java 1.ExecutorService四种线程池的例子与说明
1.new Thread的弊端 执行一个异步任务你还只是如下new Thread吗? new Thread(new Runnable() { @Override public void run() { ...
- JAVA中的4种线程池的使用
Java通过Executors提供四种线程池,分别为:newCachedThreadPool创建一个可缓存线程池,如果线程池长度超过处理需要,可灵活回收空闲线程,若无可回收,则新建线程.newFixe ...
- Java 中的几种线程池,你之前用对了吗
好久不发文章了,难道是因为忙,其实是因为懒.这是一篇关于线程池使用和基本原理的科普水文,如果你经常用到线程池,不知道你的用法标准不标准,是否有隐藏的 OOM 风险.不经常用线程池的同学,还有对几种线程 ...
- Java 中的几种线程池这么用才是对的
为什么要使用线程池 虽然大家应该都已经很清楚了,但还是说一下.其实归根结底最主要的一个原因就是为了提高性能. 线程池和数据库连接池是同样的道理,数据库连接池是为了减少连接建立和释放带来的性能开销.而线 ...
- Java 四种线程池newCachedThreadPool,newFixedThreadPool,newScheduledThreadPool,newSingleThreadExecutor
介绍new Thread的弊端及Java四种线程池的使用,对Android同样适用.本文是基础篇,后面会分享下线程池一些高级功能. 1.new Thread的弊端执行一个异步任务你还只是如下new T ...
- Java四种线程池
Java四种线程池newCachedThreadPool,newFixedThreadPool,newScheduledThreadPool,newSingleThreadExecutor 时间:20 ...
- Java四种线程池newCachedThreadPool,newFixedThreadPool,newScheduledThreadPool,newSingleThreadExecutor
1.new Thread的弊端 执行一个异步任务你还只是如下new Thread吗? Java new Thread(new Runnable() { @Override public void ru ...
- JAVA四种线程池实例
1.new Thread的弊端 执行一个异步任务你还只是如下new Thread吗? Java 1 2 3 4 5 6 7 new Thread(new Runnable() { ...
- Java 四种线程池的用法分析
1.new Thread的弊端 执行一个异步任务你还只是如下new Thread吗? new Thread(new Runnable() { @Override public void run() { ...
随机推荐
- 如何判断kbmMWClientQuery当前记录的增改状态?
有朋友问我,客户端使用了kbmMWClientQuery,对其进行了编辑后,对于指定的记录,如何判断是否是增加的记录,或者是被修改后的记录? 下面这个函数,返回aDataSet当前记录的修改状态: f ...
- Qt5配置winpCap
在网上查了很多资料,搞了差不多一天总算解决Qt5使用winPcap配置的问题了!记录一下 以便后续忘记 1.下载winpcap4.1.3,百度即可搜索到 2.下载winpCap开发者工具包http:/ ...
- C# 基础 字符串 集合 文件操作
using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.T ...
- KMS激活的密钥
今天办公电脑黑屏了,仔细一看,变成了未激活.于是从网上找到了所有Windows版本的GVLK密钥,试了一下完美激活,应该是180天的吧.先用着,等下次过期了再说,哈哈. GVLK密钥是专门用于KMS激 ...
- Win7系统打开防火墙出现0x6D9错误的解决方法
防火墙是Windows系统内的一道屏障,开启防火墙可以对系统起到一定的保护作用,可以说非常重要.但是有些Win7系统用户在开启防火墙时会被系统提示出现0x6D9的错误代码,从而不能打开防火墙. 当我们 ...
- Oracle笔记(九) 表的创建及管理
对于数据库而言实际上每一张表都表示的是一个数据库的对象,而数据库对象指的就是DDL定义的所有操作,例如:表.视图.索引.序列.约束等等,都属于对象的操作,所以表的建立就是对象的建立,而对象的操作主要分 ...
- nohup命令(Linux终端启动后台运行程序)
1. nohup指令基本概念: nohup: 不挂断的运行,并没有后台运行功能,用nohup运行命令可以使命令永久执行下去,和用户终端没有关系,断开SSH不影响运行,&是后台运行. nohup ...
- 网络初级篇之STP(BPDU详解与STP故障恢复)
一.BPDU包含的参数 通过STP的原理,我们学习了红色部分的字段(根桥ID.根路径开销.桥ID.端口ID).现在讲解一下黄色部分的字段(类型.计时器.老化时间.访问时间) 1.(BPDU Type) ...
- sql like 拼接字符串模糊查询
这种分割的值大家常用,如果要用like 来查询包含2,这个值的数据有哪些,这个怎么查? like '%2%' ????,这是不行的如果是 44,125,687 同样可以查出来,那么就想到通配符, l ...
- QTP(16)
一.QTP项目(ECShop) 1.ECShop是一个开源免费的一个B2C的电子商务系统,主要用于商家和顾客进行商品交易操作. 2.ECShop分为前台和后台两个子系统: (1)ECShop前台:顾客 ...