• Java中使用Hutool的ExecutorBuilder实现自定义线程池


    场景

    Java中ExecutorService线程池的使用(Runnable和Callable多线程实现):

    Java中ExecutorService线程池的使用(Runnable和Callable多线程实现)_霸道流氓气质的博客-CSDN博客_executorservice 设置线程池

    SpringBoot+Lombok+Builder实现任意个数属性的对象构造:

    SpringBoot+Lombok+Builder实现任意个数属性的对象构造_霸道流氓气质的博客-CSDN博客

    上面讲了自定义线程池的使用以及Builder构造者模式的一个简单应用。

    是否可以将两者结合使用,Hutool中已经实现了这一点。

    文档说明

    Hutool参考文档

    自定义线程池ExecutorBuilder

    在JDK中,提供了Executors用于创建自定义的线程池对象ExecutorService,但是考虑到线程池中存在众多概念,这些概念通过不同的搭配实现灵活的线程管理策略,单独使用Executors无法满足需求,构建了ExecutorBuilder。

    概念

    corePoolSize 初始池大小
    maxPoolSize 最大池大小(允许同时执行的最大线程数)
    workQueue 队列,用于存在未执行的线程
    handler 当线程阻塞(block)时的异常处理器,所谓线程阻塞即线程池和等待队列已满,无法处理线程时采取的策略

    线程池对待线程的策略

    1、如果池中任务数

    2、如果池中任务数>corePoolSize,放入队列等待。

    3、队列满,新建线程立即执行。

    4、执行中的线程>maxPoolSize,触发handler(RejectedExecutionHandler)异常。

    workQuene线程池策略

    SynchronousQueue它将任务直接提交给线程而不保持他们。当运行线程小于maxPoolSize时会创建新线程,否则触发异常策略。

    LinkedBlockingQueue默认无界队列,当运行线程大于corePoolSize时始终放入此队列,此时maxPoolSize无效。当构造LinkedBlockingQuene

    对象时传入参数,变成有界队列,队列满时,运行线程小于maxPoolSize时会创建新线程,否则触发异常策略。

    ArrayBlockingQuene有界队列,相对无界队列有利于控制队列大小,队列满时,运行线程小于maxPoolSize时会创建新线程,否则触发异常策略。

    注:

    博客:
    霸道流氓气质的博客_CSDN博客-C#,架构之路,SpringBoot领域博主
    关注公众号
    霸道的程序猿
    获取编程相关电子书、教程推送与免费下载。

    实现

    1、使用示例-默认线程池

    ExecutorService executor = ExecutorBuilder builder = ExecutorBuilder.create()..build();

    初始线程数为corePoolSize指定的大小
    没有最大线程数限制
    默认使用LinkedBlockingQueue,默认队列大小为1024(最大等待数1024)
    当运行线程大于corePoolSize放入队列,队列满后抛出异常

    2、使用示例-单线程线程池

    1. ExecutorService executor = ExecutorBuilder.create()//
    2.     .setCorePoolSize(1)//
    3.     .setMaxPoolSize(1)//
    4.     .setKeepAliveTime(0)//
    5.     .build();

    初始线程数为 1
    最大线程数为 1
    默认使用LinkedBlockingQueue,默认队列大小为1024
    同时只允许一个线程工作,剩余放入队列等待,等待数超过1024报错

    3、使用示例-更多选项

    1. ExecutorService executor = ExecutorBuilder.create()
    2.     .setCorePoolSize(5)
    3.     .setMaxPoolSize(10)
    4.     .setWorkQueue(new LinkedBlockingQueue<>(100))
    5.     .build();

    初始5个线程
    最大10个线程
    有界等待队列,最大等待数是100

    4、使用示例-特殊策略

    1. ExecutorService executor = ExecutorBuilder.create()
    2.     .setCorePoolSize(5)
    3.     .setMaxPoolSize(10)
    4.     .useSynchronousQueue()
    5.     .build();

    初始5个线程
    最大10个线程
    它将任务直接提交给线程而不保持它们。当运行线程小于maxPoolSize时会创建新线程,否则触发异常策略

    5、业务中使用示例

    引入项目依赖

    1.         <dependency>
    2.             <groupId>cn.hutool</groupId>
    3.             <artifactId>hutool-all</artifactId>
    4.             <version>5.8.3</version>
    5.         </dependency>

    使用示例

    1.     private static ExecutorService pool = ExecutorBuilder.create()
    2.             .setCorePoolSize(20)//初始20个线程
    3.             .setMaxPoolSize(40)//最大40个线程
    4.             .setWorkQueue(new LinkedBlockingQueue<>(60))//有界等待队列,最大等待数是60
    5.             .setThreadFactory(ThreadFactoryBuilder.create().setNamePrefix("IM-Pool-").build())//设置线程前缀
    6.             .build();

    完整使用示例

    1. package com.chrisf.imextend;
    2. import cn.hutool.core.thread.ExecutorBuilder;
    3. import cn.hutool.core.thread.ThreadFactoryBuilder;
    4. import com.chrisf.sdk.protocal.Protocal;
    5. import com.chrisf.sdk.utils.LocalSendHelper;
    6. import io.netty.channel.Channel;
    7. import java.util.Collection;
    8. import java.util.Map;
    9. import java.util.Set;
    10. import java.util.concurrent.ConcurrentHashMap;
    11. import java.util.concurrent.ExecutorService;
    12. import java.util.concurrent.LinkedBlockingQueue;
    13. /**
    14.  * im 客户端用户集
    15.  *
    16.  * @author yansh
    17.  */
    18. public class ImUsers {
    19.     private static ExecutorService pool = ExecutorBuilder.create()
    20.             .setCorePoolSize(20)//初始20个线程
    21.             .setMaxPoolSize(40)//最大40个线程
    22.             .setWorkQueue(new LinkedBlockingQueue<>(60))//有界等待队列,最大等待数是60
    23.             .setThreadFactory(ThreadFactoryBuilder.create().setNamePrefix("IM-Pool-").build())//设置线程前缀
    24.             .build();
    25.     /**
    26.      * 用户集
    27.      */
    28.     private static Map<String, Channel> USERS = new ConcurrentHashMap<String, Channel>();
    29.     /**
    30.      * 存储用户
    31.      *
    32.      * @param key     唯一键
    33.      * @param session 用户信息
    34.      */
    35.     public static void put(String key, Channel session) {
    36.         USERS.put(key, session);
    37.     }
    38.     /**
    39.      * 移除用户
    40.      *
    41.      * @param session 用户信息
    42.      * @return 移除结果
    43.      */
    44.     public static boolean remove(Channel session) {
    45.         String key = null;
    46.         boolean flag = USERS.containsValue(session);
    47.         if (flag) {
    48.             Set<Map.Entry<String, Channel>> entries = USERS.entrySet();
    49.             for (Map.Entry<String, Channel> entry : entries) {
    50.                 Channel value = entry.getValue();
    51.                 if (value.equals(session)) {
    52.                     key = entry.getKey();
    53.                     break;
    54.                 }
    55.             }
    56.         } else {
    57.             return true;
    58.         }
    59.         return remove(key);
    60.     }
    61.     /**
    62.      * 移出用户
    63.      *
    64.      * @param key
    65.      */
    66.     public static boolean remove(String key) {
    67.         Channel remove = USERS.remove(key);
    68.         if (remove != null) {
    69.             boolean containsValue = USERS.containsValue(remove);
    70.             return containsValue;
    71.         } else {
    72.             return true;
    73.         }
    74.     }
    75.     /**
    76.      * 获取在线用户列表
    77.      *
    78.      * @return 返回用户集合
    79.      */
    80.     public static Map<String, Channel> getUsers() {
    81.         return USERS;
    82.     }
    83.     /**
    84.      * 群发消息文本消息
    85.      *
    86.      * @param protocal 消息内容
    87.      */
    88.     public static void sendMessageToUsersByText(Protocal protocal) {
    89.         Collection<Channel> values = USERS.values();
    90.         for (Channel value : values) {
    91.             pool.submit(() -> sendMessageToUserByText(value, protocal));
    92.         }
    93.     }
    94.     /**
    95.      * 发送消息
    96.      *
    97.      * @param session
    98.      * @param protocal
    99.      */
    100.     public static void sendMessageToUserByText(Channel session, Protocal protocal) {
    101.         if (session != null) {
    102.             synchronized (session) {
    103.                 try {
    104.                     LocalSendHelper.sendData(session,protocal,null);
    105.                 } catch (Exception e) {
    106.                     e.printStackTrace();
    107.                 }
    108.             }
    109.         } else {
    110.         }
    111.     }
    112. }

  • 相关阅读:
    Qt的QChartView实现缩放和放大功能
    保障通航桥梁安全,创新边缘计算技术助力桥梁主动防撞预警系统
    CS61B spring21 lab7代码
    高性能AC算法多关键词匹配文本功能Java实现
    SpringBoot@Profile详解
    水文-ts中对中文进行base64编码
    传奇列表获取失败与登录器太老怎么解决
    基于ssm的图书(借阅)管理系统
    向NS-3添加新模块_ns3.35添加新模块_ns3.35以及更早版本添加新模块
    ESP32设备驱动-I2C-LCD1602显示屏驱动
  • 原文地址:https://blog.csdn.net/BADAO_LIUMANG_QIZHI/article/details/126990543