JDK自带线程池

线程池的状态

线程有如下状态

  • RUNNING状态:Accept new tasks and process queued tasks
  • SHUTDOWN状态:Don't accept new tasks, but process queued tasks
  • STOP状态: Don't accept new tasks, don't process queued tasks, and interrupt in-progress tasks
  • TIDYING状态:All tasks have terminated, workerCount is zero, the thread transitioning to state TIDYING will run the terminated() hook method
  • TERMINATED状态:terminated() has completed The numerical order among these values matters, to allow ordered comparisons.
    private final AtomicInteger ctl = new AtomicInteger(ctlOf(RUNNING, 0));
private static final int COUNT_BITS = Integer.SIZE - 3;
private static final int CAPACITY = (1 << COUNT_BITS) - 1; // runState is stored in the high-order bits
private static final int RUNNING = -1 << COUNT_BITS; // -1为全1,所以我们左移29位就是111开头的状态位
private static final int SHUTDOWN = 0 << COUNT_BITS; // 000
private static final int STOP = 1 << COUNT_BITS; // 001
private static final int TIDYING = 2 << COUNT_BITS; // 010
private static final int TERMINATED = 3 << COUNT_BITS; // 011 // Packing and unpacking ctl
private static int runStateOf(int c) { return c & ~CAPACITY; }
private static int workerCountOf(int c) { return c & CAPACITY; }
private static int ctlOf(int rs, int wc) { return rs | wc; }

我们从上面的源代码中可以看出,我们将状态存储在一个原子整型的前三位,然后将线程的容量存储在后29位。将状态和容量放在一起,这样更新状态和容量只需要进行一次cas操作。

下面的三个方法就是进行获取状态和工作线程数量和初始化状态。

在源代码注释中解释到,当未来原子整型不够用了,就会将其升级为原子长整形。且状态位也有扩展的空间,如果需要的话。

同时源代码中也有表明各个状态转换的条件,可以ThreadPoolExecutor类中下载source查看。

线程池构造方法

参数组成

  • corePoolSize核心线程的数量
  • maximumPoolSize最大的线程数量 PS: 最大线程数-核心线程数 = 急救线程的数量
  • keepAliveTime 急救线程的存活时间
  • unit 急救线程存活时间单位
  • workQueue 阻塞队列
  • threadFactory 线程工厂 PS:线程工厂就是创造线程的工厂,为其进行给任务和名字
  • handler 拒绝策略的实现 PS:就是当阻塞队列满了之后所要做的动作,死等,限时等(RocketMQ),交给调用者运行,直接抛弃,创建一个新线程(netty),抛出异常写日志(dubbo)。
    public ThreadPoolExecutor(int corePoolSize,
int maximumPoolSize,
long keepAliveTime,
TimeUnit unit,
BlockingQueue<Runnable> workQueue,
ThreadFactory threadFactory,
RejectedExecutionHandler handler)

这就是拒绝策略在JDK自带的实现。

  • AbortPolicy直接抛出一个RejectedExecutionException异常,dubbo应该是加以记录一些更多的信息 猜测
  • CallerRunsPolicy就是让调用者自己执行这个任务
  • DiscardPolicy直接抛弃
  • DiscardOldestPolicy抛弃早进入阻塞队列的然后让当前任务进入阻塞队列

JDK线程池和上一次的线程池不一样的就是急救线程。

急救线程就是当阻塞队列满了之后,并不会像我上次的例子一样直接进行拒绝策略的判断,会创建一个急救线程或者已存活的急救线程进行执行任务,如果急救线程也满了的话,才会进入拒绝策略的判断。

JDK线程池的基本使用

固定大小线程池

ExecutorService executorService = Executors.newFixedThreadPool(2, new ThreadFactory() {
AtomicInteger ctl = new AtomicInteger(); @Override
public Thread newThread(Runnable r) {
Thread thread = new Thread(r,"myThreadPoll-" + ctl.getAndIncrement());
return thread;
}
});
executorService.execute(()->{
log.debug("线程执行了一次");
});
executorService.execute(()->{
log.debug("线程执行了一次");
});
executorService.execute(()->{
log.debug("线程执行了一次");
});

我看了一下它的默认构造方法,它创造了一个Integer.MAX_VALUE大小的阻塞队列。阻塞队列其实和小测验的阻塞队列是差不多的。

缓存线程池

主要是它的阻塞队列的不同,其中核心数为0,然后通过阻塞队列直到有线程对其进行取任务,不然就是一直阻塞的状态。

@Slf4j
public class Test2 { public static void main(String[] args) {
ExecutorService executorService = Executors.newCachedThreadPool();
executorService.execute(()->{
log.debug("执行成功");
});
executorService.execute(()->{
log.debug("执行成功");
});
} }

A blocking queue in which each insert operation must wait for a corresponding remove operation by another thread, and vice versa.

翻译:一种阻塞队列,其中每个插入操作必须等待另一个线程执行相应的删除操作,反之亦然。

单线程线程池

顾名思义即单线程的线程池,不过有意思的一点就是这个使用了一个设计模式就是装饰器模式。

@Slf4j
public class Test3 { public static void main(String[] args) {
ExecutorService executorService = Executors.newSingleThreadExecutor();
executorService.execute(()->{
log.debug("hello");
});
executorService.execute(()->{
log.debug("hello");
});
} }

首先是因为如果我们直接返回ThreadPoolExecutor这个类的话,我们是知道了它的类,是可以直接使用强转来实现修改线程的核心数以及一些参数。如下

@Slf4j
public class Test1 {
public static void main(String[] args) {
ExecutorService executorService = Executors.newFixedThreadPool(2, new ThreadFactory() {
AtomicInteger ctl = new AtomicInteger(); @Override
public Thread newThread(Runnable r) {
Thread thread = new Thread(r,"myThreadPoll-" + ctl.getAndIncrement());
return thread;
}
});
// 我们将其对象通过强转直接修改了其的核心数,执行结构同样生效
ThreadPoolExecutor executor= (ThreadPoolExecutor) executorService;
executor.setCorePoolSize(1);
executorService.execute(()->{
log.debug("线程执行了一次");
});
executorService.execute(()->{
log.debug("线程执行了一次");
});
executorService.execute(()->{
log.debug("线程执行了一次");
});
}
}

但是如果我们通过装饰器模式将其进行包装,然后包装的对象返回,是无法进行修改核心数的,更何况单线程线程池的情况下,我们需要保证核心数总为1把,不能让其他人修改。展示部分代码,返回的是其的装饰类。

static class DelegatedExecutorService extends AbstractExecutorService {
private final ExecutorService e;
DelegatedExecutorService(ExecutorService executor) { e = executor; }
public void execute(Runnable command) { e.execute(command); }
public void shutdown() { e.shutdown(); }
public List<Runnable> shutdownNow() { return e.shutdownNow(); }

ExecutorService方法使用

这个类就是线程池的接口类,掌管着线程池的方法。

  • void shutdown() 继续执行当前线程池中的任务和阻塞队列中的任务,不再接收新的任务。
  • List shutdownNow() 尝试停止所有正在执行的任务,停止正在等待的任务的处理,并返回正在等待执行的任务的列表
  • boolean isShutdown() 返回当前线程池是否已经关闭
  • boolean isTerminated() 在调用了shutdown或Shutdownow后,关闭后所有任务都已完成,则返回true。如果没有调用永远不会返回true。
  • boolean awaitTermination(long timeout, TimeUnit unit) 阻塞,直到所有任务在关闭请求后完成执行,或超时发生,或当前线程中断,以先发生的为准。
  • Future submit(Callable task) 返回执行结果
  • List<Future> invokeAll(Collection<? extends Callable> tasks) 批量返回结果
  • T invokeAny(Collection<? extends Callable> tasks) 执行任意一个并返回结果,如果任何一个完成,其他正在执行的直接结束。

工作线程的饥饿现象

/**
* @Author 10276
* @Date 2022/5/11 20:52
*/
@Slf4j
public class Test4 { public static void main(String[] args) {
ExecutorService rest = Executors.newFixedThreadPool(2);
rest.execute(()->{
log.debug("准备点餐");
Future<String> submit = rest.submit(() -> {
log.debug("正在坐菜");
return "菜";
});
try {
log.debug("上菜{}",submit.get());
} catch (InterruptedException | ExecutionException e) {
e.printStackTrace();
}
});
rest.execute(()->{
log.debug("准备点餐");
Future<String> submit = rest.submit(() -> {
log.debug("正在坐菜");
return "菜";
});
try {
log.debug("上菜{}",submit.get());
} catch (InterruptedException | ExecutionException e) {
e.printStackTrace();
}
});
}
}

一个线程池的导致两个线程没办法继续进行下去,没有多余的线程取做接下来的事情,但这不是死锁问题,归结原因还是线程资源不够,同时也无法继续进行下去了。

解决方案:由此可以得出对于线程池数量的选择和线程池中核心线程数量的选择是十分重要的。

public static void main(String[] args) {
ExecutorService waitress = Executors.newFixedThreadPool(1);
ExecutorService cooker = Executors.newFixedThreadPool(1);
waitress.execute(()->{
log.debug("准备点餐");
Future<String> submit = cooker.submit(() -> {
log.debug("正在做菜");
return "湖南菜";
});
try {
log.debug("上菜:{}",submit.get());
} catch (InterruptedException | ExecutionException e) {
e.printStackTrace();
}
});
waitress.execute(()->{
log.debug("准备点餐");
Future<String> submit = cooker.submit(() -> {
log.debug("正在做菜");
return "广东菜";
});
try {
log.debug("上菜:{}",submit.get());
} catch (InterruptedException | ExecutionException e) {
e.printStackTrace();
}
});
}

JDK自带线程池学习的更多相关文章

  1. JDK自带线程池介绍及使用环境

    1.newFixedThreadPool创建一个指定工作线程数量的线程池.每当提交一个任务就创建一个工作线程,如果工作线程数量达到线程池初始的最大数,则将提交的任务存入到池队列中. 2.newCach ...

  2. Java线程池学习

    Java线程池学习 Executor框架简介 在Java 5之后,并发编程引入了一堆新的启动.调度和管理线程的API.Executor框架便是Java 5中引入的,其内部使用了线程池机制,它在java ...

  3. 【Java多线程】线程池学习

    Java线程池学习 众所周知,Java不仅提供了线程,也提供了线程池库给我们使用,那么今天来学学线程池的具体使用以及线程池基本实现原理分析. ThreadPoolExecutor ThreadPool ...

  4. Java高并发程序设计学习笔记(六):JDK并发包(线程池的基本使用、ForkJoin)

    转自:https://blog.csdn.net/dataiyangu/article/details/86573222 1. 线程池的基本使用1.1. 为什么需要线程池1.2. JDK为我们提供了哪 ...

  5. 从源码看JDK提供的线程池(ThreadPoolExecutor)

    一丶什么是线程池 (1)博主在听到线程池三个字的时候第一个想法就是数据库连接池,回忆一下,我们在学JavaWeb的时候怎么理解数据库连接池的,数据库创建连接和关闭连接是一个比较耗费资源的事情,对于那些 ...

  6. (转)java自带线程池和队列详细讲解 - CSDN过天的专栏

    一简介 线程的使用在java中占有极其重要的地位,在jdk1.4极其之前的jdk版本中,关于线程池的使用是极其简陋的.在jdk1.5之后这一情况有了很大的改观.Jdk1.5之后加入了java.util ...

  7. java自带线程池和队列详细讲解

    Java线程池使用说明 一简介 线程的使用在java中占有极其重要的地位,在jdk1.4极其之前的jdk版本中,关于线程池的使用是极其简陋的.在jdk1.5之后这一情况有了很大的改观.Jdk1.5之后 ...

  8. JAVA线程池学习,ThreadPoolTaskExecutor和ThreadPoolExecutor有何区别?

    初学者很容易看错,如果没有看到spring或者JUC源码的人肯定是不太了解的. ThreadPoolTaskExecutor是spring core包中的,而ThreadPoolExecutor是JD ...

  9. c++11 线程池学习笔记 (一) 任务队列

    学习内容来自一下地址 http://www.cnblogs.com/qicosmos/p/4772486.html github https://github.com/qicosmos/cosmos ...

随机推荐

  1. 机器学习之linear_model (线性回归算法模型)

    1.matplotlib 首先看一下这个静态图绘制模块 静态图形处理 数据分析三剑客 Numpy : 主要为了给pandas提供数据源 pandas : 更重要的数据结构 matplotlib : 静 ...

  2. IDEA问题之“微服务启动项目时,不会加载Spring Boot到Services中”

    1.启动项目时,不会加载Spring Boot到Services中 现象解析: 启动项目时 会在debug的位置加载项目 注:这里没有配图,因为问题已解决,未记录图,需往后遇到记录 解决方案: 需要在 ...

  3. JS+CSS3 360度全景图插件 - Watch3D.js

    日常闲扯 从上一篇文章到这篇中间快过了一年了,时间真滴过得快.不是在下中间没想过写新的文章,而是自己确实变懒了(体重+1 +1 +1 +1....) ..OTL...不过到最后觉得还是需要写点东西,不 ...

  4. HCNP Routing&Switching之组播技术-组播路由协议PIM

    前文我们了解了组播技术中组播分发树相关话题,回顾请参考https://www.cnblogs.com/qiuhom-1874/p/16019334.html:今天我们来聊一聊组播路由协议PIM相关话题 ...

  5. Java/C++实现备忘录模式--撤销操作

    改进课堂上的"用户信息操作撤销"实例,使得系统可以实现多次撤销(可以使用HashMap.ArrayList等集合数据结构实现). 类图: Java代码: import java.u ...

  6. idea 启动微服务 设置 run dashboard

    微服务如果很多,启动时如果在run窗口,会不是很方便,所以idea中配置了rundashboard,有时不自动出现时,需要进行配置: 配置操作如下: 我的idea版本2020.2 1.在父工程的.id ...

  7. JAVA处理Excel表格数据并写入数据库

    package com.hncj.test; import java.io.FileInputStream; import java.sql.Connection; import java.sql.D ...

  8. CCF201503-2数字排序

    问题描述 给定n个整数,请统计出每个整数出现的次数,按出现次数从多到少的顺序输出. 输入格式 输入的第一行包含一个整数n,表示给定数字的个数. 第二行包含n个整数,相邻的整数之间用一个空格分隔,表示所 ...

  9. 解决 Tomcat 控制台输出乱码(Tomcat Localhost Log / Tomcat Catalina Log 乱码)

    1. 按下图修改 先找到你的 Tomcat 安装目录,然后进入conf文件夹,找到 logging.properties,并打开它,然后把所有 UTF-8 格式的编码改成 GBK即可,具体操作如下图

  10. vux+vue-cli3.0坑

    最近开发了项目使用了vue-cli3.0+vux搭建的项目,现在总结遇到的问题: 环境:github vux有关于vue-cli3.0以及vux已经搭建好的脚手架vux-cli3链接 一:如下报错 解 ...