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. 初识Spring(为什么要使用Spring?)

    Spring,英文翻译是春天的意思,而在Java中,是一个开放源代码的设计层面框架(手动滑稽,程序员的春天),他解决的是业务逻辑层和其他各层的松耦合问题,因此它将面向接口的编程思想贯穿整个系统应用.S ...

  2. 什么是 Spring 的依赖注入?

    依赖注入,是 IOC 的一个方面,是个通常的概念,它有多种解释.这概念是说你 不用创建对象,而只需要描述它如何被创建.你不在代码里直接组装你的组件和 服务,但是要在配置文件里描述哪些组件需要哪些服务, ...

  3. django基础环境配置

    Django环境搭建 1.下载安装 命令行 pip install django==1.11.21 pip install django==1.11.21 -i 源 pycharm setting - ...

  4. 学习GlusterFS(一)

    一.概述 1.GlusterFS是集群式NAS存储系统,分布式文件系统(POSIX兼容),Tcp/Ip方式互联的一个并行的网络文件系统,通过原生 GlusterFS 协议访问数据,也可以通过 NFS/ ...

  5. CSDN博客步骤:

    在SCDN看到喜欢的文章想转载又嫌一个一个敲太麻烦,干脆直接收藏.但有时候作者把原文章删除或设置为私密文章后又看不了.所以还是转载来的好.这篇博文为快速转载博客的方法,亲测有效,教程如下. 原博客原址 ...

  6. 切图崽的自我修养-[ES6] 迭代器Iterator浅析

    Iterator 这真是毅种循环 Iterator不是array,也不是set,不是map, 它不是一个实体,而是一种访问机制,是一个用来访问某个对象的接口规范,为各种不同的数据结构提供统一的访问机制 ...

  7. js 获取和设置css3 属性值的实现方法

    众多周知 CSS3 增加了很多属性,在读写的时候就没有原先那么方便了. 如:<div style="left:100px"></div> 只考虑行间样式的话 ...

  8. CSS:两端对齐原理(text-align:justify)

    我是一个小白我是一个小白我是一个小白喷我吧,哈哈 写样式的是时候经常会碰到字体两端对齐的效果,一般就网上找端css样式复制下就结束了,没有考虑过原理是啥贴下代码 <head> <me ...

  9. Java/C++实现模板方法模式---数据库操作

    对数据库的操作一般包括连接.打开.使用.关闭等步骤,在数据库操作模板类中我们定义了connDB().openDB().useDB().closeDB()四个方法分别对应这四个步骤.对于不同类型的数据库 ...

  10. 常⽤的meta标签有哪些

    meta 标签由 name 和 content 属性定义,用来描述网页文档的属性,比如网页的作者,网页描述,关键词等,除了HTTP标准固定了一些name作为大家使用的共识,开发者还可以自定义name. ...