本文分享创建线程工厂 ThreadFactory 的三种方式,以方便大家快速创建线程池,并通过线程工厂给每个创建出来的线程设置极富业务含义的名字。

线程池大小考虑因素

  由于需要自定义线程池,故这里先介绍线程池大小如何设定最为合理。我们需要分析计算环境、资源预算和任务的特性,例如,考虑一下在部署的服务器上有多少个CPU,内存是多大,任务是计算密集型、I/O密集型还是二者皆可。

  对于计算密集型的任务,在拥有N(CPU)个处理器的服务器上,当线程池的大小为N+1时,通常能实现最优的利用率。对于包含I/O操作或者其它阻塞操作的任务,由于线程并不会一直执行,因此线程池的规模应该更大,参考值可以设置为2**N(CPU)。线程池资源并不是唯一影响线程池大小的资源,还包括内存、文件句柄、套接字句柄和数据库连接等。在观察任务运行情况和系统负载、资源利用率之后,请酌情调整。

  在Java中,可以使用如下代码获取CPU的数量:

int N(CPU)= Runtime.getRuntime().availableProcessors();

基于此可以设置合适的线程池数量,提高系统稳定性和吞吐率。

Spring CustomizableThreadFactory

  此方式是利用Spring 框架提供的轮子 CustomizableThreadFactory。

ThreadFactory springFactory = new CustomizableThreadFactory("spring-pool-");
ExecutorService exec = new ThreadPoolExecutor(1, 1,
0L, TimeUnit.MILLISECONDS,
new LinkedBlockingQueue<Runnable>(100), springFactory);
exec.submit(() -> {
logger.info("--记忆中的颜色是什么颜色---");
});

guava ThreadFactoryBuilder

  使用Google 开源框架guava提供的 ThreadFactoryBuilder 可以快速给线程池里的线程设置有意义的名字。

ThreadFactory guavaFactory = new ThreadFactoryBuilder().setNameFormat("guava-pool-").build();

ExecutorService exec = new ThreadPoolExecutor(1, 1,
0L, TimeUnit.MILLISECONDS,
new LinkedBlockingQueue<Runnable>(100),guavaFactory );
exec.submit(() -> {
logger.info("--记忆中的颜色是什么颜色---");
});

此方法需要引入如下guava maven坐标:

     <dependency>
<groupId>com.google.guava</groupId>
<artifactId>guava</artifactId>
<version>20.0</version>
</dependency>

Apache commons-lang3 BasicThreadFactory

  Apache commons-lang3 提供的 BasicThreadFactory.

ThreadFactory basicFactory = new BasicThreadFactory.Builder()
.namingPattern("basicFactory-pool-").build(); ExecutorService exec = new ThreadPoolExecutor(1, 1,
0L, TimeUnit.MILLISECONDS,
new LinkedBlockingQueue<Runnable>(100),basicFactory );
exec.submit(() -> {
logger.info("--记忆中的颜色是什么颜色---");
});

  使用的列队是无界队列LinkedBlockingQueue。如果当前执行任务数量大于核心线程数,此时再提交任务就在添加到阻塞队列中等待执行,直到有可用线程。温馨提示,无界阻塞队列可能消耗很大的内存资源。值得注意的是,如果使用了无界的任务队列,线程池最大数量这个参数就没什么效果了。

优雅的自定义线程工厂

  除了使用第三方jar包之外,我们也可以基于ThreadPoolExecutor优雅地自定义ThreadFactory,并根据自己的需要来操作线程,下面是实例代码:

import com.google.common.util.concurrent.ThreadFactoryBuilder;
import lombok.extern.slf4j.Slf4j; import java.util.concurrent.*; @Slf4j
public class ThreadPoolStudy {
public static void main(String[] args) {
ExecutorService executor = initExecutor();
// 若i>0,则不会抛出异常信息。设为0是为了验证如何捕捉异常
for (int i = 0; i < 10; i++) {
DivTask myTask = new DivTask(100, i);
Future future = executor.submit(myTask);
try {
// 阻塞方法,尽量不要调用,这里调用get是为了避免任务执行异常被吞掉
future.get();
} catch (Exception e) {
log.error("任务执行异常,", e);
}
}
executor.shutdown();
log.info("线程池马上关闭,不再接收新任务。");
}
// 自定义任务,用于求两数相除的结果
class DivTask implements Runnable {
int a, b; public DivTask(int a, int b) {
this.a = a;
this.b = b;
} @Override
public void run() {
log.info("计算结果:{}", a / b);
}
} /**
* 初始化线程池
*
* @return
*/
private static ExecutorService initExecutor() {
ThreadFactory guavaFactory = new ThreadFactoryBuilder().setNameFormat("guava-pool-").build();
ExecutorService executor = new ThreadPoolExecutor(1, 64,
30L, TimeUnit.MILLISECONDS,
new SynchronousQueue<Runnable>(), guavaFactory); return executor;
}
}

  任务执行完毕后,请关闭线程池,及时降低资源损耗。

小结

  作为程序员,要有“刨根问底”的精神。知其然,更要知其所以然。这篇文章希望能帮助你对自定义线程工厂抽丝剥茧,还原背后的原理。

Reference

Java 自定义线程池的线程工厂的更多相关文章

  1. 由浅入深理解Java线程池及线程池的如何使用

    前言 多线程的异步执行方式,虽然能够最大限度发挥多核计算机的计算能力,但是如果不加控制,反而会对系统造成负担.线程本身也要占用内存空间,大量的线程会占用内存资源并且可能会导致Out of Memory ...

  2. -1-5 java 多线程 概念 进程 线程区别联系 java创建线程方式 线程组 线程池概念 线程安全 同步 同步代码块 Lock锁 sleep()和wait()方法的区别 为什么wait(),notify(),notifyAll()等方法都定义在Object类中

     本文关键词: java 多线程 概念 进程 线程区别联系 java创建线程方式 线程组 线程池概念 线程安全 同步 同步代码块 Lock锁  sleep()和wait()方法的区别 为什么wait( ...

  3. Java多线程、线程池和线程安全整理

    多线程 1.1      多线程介绍 进程指正在运行的程序.确切的来说,当一个程序进入内存运行,即变成一个进程,进程是处于运行过程中的程序,并且具有一定独立功能. 1.2      Thread类 通 ...

  4. Java多线程系列 JUC线程池02 线程池原理解析(一)

    转载  http://www.cnblogs.com/skywang12345/p/3509960.html ; http://www.cnblogs.com/skywang12345/p/35099 ...

  5. 适配器、工厂模式、线程池、线程组、互斥锁、Timer类、Runtime类、单例设计模式(二十四)

    1.多线程方法 * Thread 里面的俩个方法* 1.yield让出CPU,又称为礼让线程* 2.setPriority()设置线程的优先级 * 优先级最大是10,Thread.MAX_PRIORI ...

  6. java 线程池(线程的复用)

    一. 线程池简介 1. 线程池的概念: 线程池就是首先创建一些线程,它们的集合称为线程池.使用线程池可以很好地提高性能,线程池在系统启动时即创建大量空闲的线程,程序将一个任务传给线程池,线程池就会启动 ...

  7. Java多线程系列 JUC线程池03 线程池原理解析(二)

    转载  http://www.cnblogs.com/skywang12345/p/3509954.html  http://www.cnblogs.com/skywang12345/p/351294 ...

  8. Java多线程系列 JUC线程池01 线程池框架

    转载  http://www.cnblogs.com/skywang12345/p/3509903.html 为什么引入Executor线程池框架 new Thread()的缺点 1. 每次new T ...

  9. Java多线程系列 JUC线程池07 线程池原理解析(六)

     关闭“线程池” shutdown()的源码如下: public void shutdown() { final ReentrantLock mainLock = this.mainLock; // ...

  10. 多线程、线程池、线程创建、Thread

    转载自https://www.cnblogs.com/jmsjh/p/7762034.html 多线程 1.1 多线程介绍 学习多线程之前,我们先要了解几个关于多线程有关的概念. 进程:进程指正在运行 ...

随机推荐

  1. 从文件到块: 提高 Hugging Face 存储效率

    Hugging Face 在 Git LFS 仓库 中存储了超过 30 PB 的模型.数据集和 Spaces.由于 Git 在文件级别进行存储和版本控制,任何文件的修改都需要重新上传整个文件.这在 H ...

  2. Selenium KPI接口 附件上传

    实现功能 拖拽图片到百度上传图片搜索功能区域. 定位.send_keys(r'图片路径') 导入相关包 from selenium import webdriver from time import ...

  3. 移动端 cordova vue videojs 全屏播放后退出全屏返回后退出app问题

    问题描述 移动端上面使用了videojs 播放视频,同时也监听了手机返回事件document.addEventListener('backbutton',.接着我们点击全屏播放后在退出全屏在返回后直接 ...

  4. Arrays工具类教你优雅地管理数组数据

    数组专用工具类指的是 java.util.Arrays 类,基本上常见的数组操作,这个类都提供了静态方法可供直接调用.毕竟数组本身想完成这些操作还是挺麻烦的,有了这层封装,就方便多了. package ...

  5. MAMP使用简单教程

    这个配置,没有域名访问,平时可以放些demo使用,如果需要域名访问请看MAMP PRO教程 启用服务 打开Launchpad中灰色的MAMP,进入界面后,点击Preferences,然后只需拿着鼠标点 ...

  6. linux命令:lsof命令

    lsof(list open files)是一个列出当前系统打开文件的工具.在linux环境下,任何事物都以文件的形式存在,通过文件不仅仅可以访问常规数据,还可以访问网络连接和硬件.所以如传输控制协议 ...

  7. [每日算法 - 华为机试] leetcode690. 员工的重要性

    入口 力扣(LeetCode)官网 - 全球极客挚爱的技术成长平台备战技术面试?力扣提供海量技术面试资源,帮助你高效提升编程技能,轻松拿下世界 IT 名企 Dream Offer.https://le ...

  8. Anonymous打靶学习笔记(5)

    参考视频https://www.bilibili.com/video/BV1XufaYAEKc/?spm_id_from=333.1387.search.video_card.click 常见的100 ...

  9. Eclipse 中 JAVA AWT相关包不提示问题(解决)

    原因: 由于在2021年7月15日 OpenJDK管理委员会全票通过批准成立由Phil Race担任初始负责人的 Client Libraries Group(客户端类库工作组). 新的工作组将继续赞 ...

  10. Sentinel源码—9.限流算法的实现对比

    大纲 1.漏桶算法的实现对比 (1)普通思路的漏桶算法实现 (2)节省线程的漏桶算法实现 (3)Sentinel中的漏桶算法实现 (4)Sentinel中的漏桶算法与普通漏桶算法的区别 (5)Sent ...