当前位置: 首页 > news >正文

自定义线程池(二)

上节回顾

在上一节当中,已经实现了一个线程池,在本节当中,我们需要添加拒绝策略。这里使用到了策略模式的设计模式,因为拒绝策略是多种的,我们需要将这个权利下放给调用者(由调用者来指定我要采取哪种策略),而线程池只需要调用拒绝的接口即可。
在这里插入图片描述

步骤

(1)定义拒绝策略接口
(2)在线程池中加入拒绝策略参数
(3)自行调用测试

1.定义接口类

@FunctionalInterface
interface RejectPolicy<T>{//注意传递参数void reject(BlockQueue<T> queue,Runnable task);
}

2.线程池中添加接口以及调用方法

@Slf4j
class ThreadPool {//任务队列private BlockQueue<Runnable> taskQueue;//线程集合 我们需要对线程做一个包装private HashSet<Worker> workers = new HashSet<>();//核心线程数量private long coreSize;//超时时间private long timeout;//时间单位private TimeUnit timeUnit;//自定义拒绝策略private RejectPolicy<Runnable> rejectPolicy;public ThreadPool(int queueCapacity,long coreSize,long timeout,TimeUnit timeUnit){this.taskQueue = new BlockQueue<>(queueCapacity);this.coreSize = coreSize;this.timeout = timeout;this.timeUnit = timeUnit;this.rejectPolicy = (queue, task) -> {throw new RuntimeException();};}public ThreadPool(int queueCapacity,long coreSize,long timeout,TimeUnit timeUnit,RejectPolicy<Runnable> rejectPolicy){taskQueue = new BlockQueue<>(queueCapacity);this.coreSize = coreSize;this.timeout = timeout;this.timeUnit = timeUnit;this.rejectPolicy = rejectPolicy;}//执行任务public void execute(Runnable task){//当任务数量尚未超过coreSizesynchronized (workers){if (workers.size() < coreSize){log.info("创建工作线程{}",task);Worker worker = new Worker(task);workers.add(worker);worker.start();}else{log.info("加入到任务队列{}",task);//有可能会阻塞在这里 进而将主线程阻塞掉//taskQueue.put(task);//这里会有很多种策略自定义策略//策略模式:操作抽象成接口实现代码是传过来不会写死taskQueue.tryPut(rejectPolicy,task);//rejectPolicy.reject(taskQueue,task);}}}class Worker extends Thread{private Runnable task;public Worker(Runnable task){this.task = task;}@Overridepublic void run() {while (task != null || (task = taskQueue.poll(timeout,timeUnit)) != null){try {log.info("正在执行...{}",task);//执行任务task.run();}catch (Exception e){System.out.println(e.getMessage());}finally {//不要忘记这一步task = null;}}synchronized (workers){log.info("worker被移除{}",this);workers.remove(this);}}}
}

3.main测试

@Slf4j
public class TestPool {//阻塞队列是平衡生产者和消费者之间的中介//任务数量超过任务队列的情况public static void main(String[] args) {ThreadPool threadPool = new ThreadPool(10, 2, 1000, TimeUnit.MICROSECONDS, (queue, task) -> {//1.死等queue.put(task);//2.超时等待queue.offer(task, 1500, TimeUnit.MICROSECONDS);//3.调用者自己放弃// log.debug("放弃{}",task);//4.调用者抛异常//throw new RuntimeException("task执行失败" + task);//5.调用者自己执行task.run();});for (int i = 0; i < 20; i++) {int j = i;//主线程可能会在这里阻塞threadPool.execute(() -> {try {Thread.sleep(30000);} catch (InterruptedException e) {throw new RuntimeException(e);}TestPool.log.debug("{}", j);});}}
}
http://www.lryc.cn/news/416492.html

相关文章:

  • 【Linux】常见指令
  • uniapp自定义网格布局用于选择金额、输入框焦点事件以及点击逻辑实战
  • 中小学创客室培养学生全面发展
  • AI Agent智能体落地应用测试,一句话即可操控它执行工作
  • 免费的SD-WAN服务
  • gradle安装及配置
  • C-sharp-console-gui-framework:C#控制台应用程序的GUI框架
  • 一文搞懂后端面试之MySQL MVCC【中间件 | 数据库 | MySQL | 隔离级别 | Read View】
  • Mysql执行计划(上)
  • 使用Python+moviepy截取音频片段
  • Java学习Day19
  • 8.达梦数据库常用SQL
  • 深入理解接口测试:实用指南与最佳实践(四)IHRM管理系统实战-项目分析
  • 程序编译及链接
  • route 命令介绍及使用方法
  • 力扣热题100_二叉树_226_翻转二叉树
  • Java SpringBoot 集成 MinIO 资料
  • 鸿蒙系统开发【加解密算法库框架】安全
  • C语言——二维数组和字符数组
  • Python 爬虫入门(九):Scrapy安装及使用「详细介绍」
  • 扩展addr2line程序的功能,group_add2line() 脚本的实现
  • idea中修改项目名称
  • Flink开发语言使用Java还是Scala合适?
  • C++STL专题 vector底层实现
  • 【Linux】装机常用配置
  • oracle库PASSWORD_VERSIONS 对应的加密方式
  • 分享一个基于微信小程序的乡村医疗上门服务预约平台(源码、调试、LW、开题、PPT)
  • 切香肠(Sausage)
  • Session与Cookie以及Cache区别,及应用场景
  • Debian | 更换 Gnome 至 Xfce4