自定义线程池(二)

打印 上一主题 下一主题

主题 830|帖子 830|积分 2492

上节回顾

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

步骤

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

  1. @FunctionalInterface
  2. interface RejectPolicy<T>{
  3.     //注意传递参数
  4.     void reject(BlockQueue<T> queue,Runnable task);
  5. }
复制代码
2.线程池中添加接口以及调用方法

  1. @Slf4j
  2. class ThreadPool {
  3.     //任务队列
  4.     private BlockQueue<Runnable> taskQueue;
  5.     //线程集合 我们需要对线程做一个包装
  6.     private HashSet<Worker> workers = new HashSet<>();
  7.     //核心线程数量
  8.     private long coreSize;
  9.     //超时时间
  10.     private long timeout;
  11.     //时间单位
  12.     private TimeUnit timeUnit;
  13.     //自定义拒绝策略
  14.     private RejectPolicy<Runnable> rejectPolicy;
  15.     public ThreadPool(int queueCapacity,long coreSize,long timeout,TimeUnit timeUnit){
  16.         this.taskQueue = new BlockQueue<>(queueCapacity);
  17.         this.coreSize = coreSize;
  18.         this.timeout = timeout;
  19.         this.timeUnit = timeUnit;
  20.         this.rejectPolicy = (queue, task) -> {
  21.             throw new RuntimeException();
  22.         };
  23.     }
  24.     public ThreadPool(int queueCapacity,long coreSize,long timeout,TimeUnit timeUnit,RejectPolicy<Runnable> rejectPolicy){
  25.         taskQueue = new BlockQueue<>(queueCapacity);
  26.         this.coreSize = coreSize;
  27.         this.timeout = timeout;
  28.         this.timeUnit = timeUnit;
  29.         this.rejectPolicy = rejectPolicy;
  30.     }
  31.     //执行任务
  32.     public void execute(Runnable task){
  33.         //当任务数量尚未超过coreSize
  34.         synchronized (workers){
  35.             if (workers.size() < coreSize){
  36.                 log.info("创建工作线程{}",task);
  37.                 Worker worker = new Worker(task);
  38.                 workers.add(worker);
  39.                 worker.start();
  40.             }else{
  41.                 log.info("加入到任务队列{}",task);
  42.                 //有可能会阻塞在这里 进而将主线程阻塞掉
  43.                 //taskQueue.put(task);
  44.                 //这里会有很多种策略自定义策略
  45.                 //策略模式:操作抽象成接口实现代码是传过来不会写死
  46.                 taskQueue.tryPut(rejectPolicy,task);
  47.                 //rejectPolicy.reject(taskQueue,task);
  48.             }
  49.         }
  50.     }
  51.     class Worker extends Thread{
  52.         private Runnable task;
  53.         public Worker(Runnable task){
  54.             this.task = task;
  55.         }
  56.         @Override
  57.         public void run() {
  58.            while (task != null || (task = taskQueue.poll(timeout,timeUnit)) != null){
  59.                try {
  60.                    log.info("正在执行...{}",task);
  61.                    //执行任务
  62.                    task.run();
  63.                }catch (Exception e){
  64.                    System.out.println(e.getMessage());
  65.                }finally {
  66.                    //不要忘记这一步
  67.                    task = null;
  68.                }
  69.            }
  70.            synchronized (workers){
  71.                log.info("worker被移除{}",this);
  72.                workers.remove(this);
  73.            }
  74.         }
  75.     }
  76. }
复制代码
3.main测试

  1. @Slf4j
  2. public class TestPool {
  3.     //阻塞队列是平衡生产者和消费者之间的中介
  4.     //任务数量超过任务队列的情况
  5.     public static void main(String[] args) {
  6.         ThreadPool threadPool = new ThreadPool(10, 2, 1000, TimeUnit.MICROSECONDS, (queue, task) -> {
  7.             //1.死等
  8.             queue.put(task);
  9.             //2.超时等待
  10.             queue.offer(task, 1500, TimeUnit.MICROSECONDS);
  11.             //3.调用者自己放弃
  12.             // log.debug("放弃{}",task);
  13.             //4.调用者抛异常
  14.             //throw new RuntimeException("task执行失败" + task);
  15.             //5.调用者自己执行
  16.             task.run();
  17.         });
  18.         for (int i = 0; i < 20; i++) {
  19.             int j = i;
  20.             //主线程可能会在这里阻塞
  21.             threadPool.execute(() -> {
  22.                 try {
  23.                     Thread.sleep(30000);
  24.                 } catch (InterruptedException e) {
  25.                     throw new RuntimeException(e);
  26.                 }
  27.                 TestPool.log.debug("{}", j);
  28.             });
  29.         }
  30.     }
  31. }
复制代码
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

x
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

刘俊凯

金牌会员
这个人很懒什么都没写!

标签云

快速回复 返回顶部 返回列表