• 编码规约对于多线程的使用解读


    起因:同事代码手动创建线程

    对同事的一段代码进行了优化

    部署后,规约扫描报错“不能手动创建线程池”。

    3.【强制】 线程资源必须通过线程池提供,不允许在应用中自行显式创建线程。

    说明:使用线程池的好处是减少在创建和销毁线程上所花的时间以及系统资源的开销,解决资源不足的问题。如果不使用线程池,有可能造成系统创建大量同类线程而导致消耗完内存或者“过度切换”的问题。

    修改1:使用Executors创建线程池

    1. ExecutorService threadPool = Executors.newCachedThreadPool(); //创建大小可伸缩的线程池
    2. try {
    3. for (int i = 0; i < 30; i++) {
    4. //使用线程池来创建线程
    5. threadPool.execute(()->{
    6. System.out.println(Thread.currentThread().getName()+"ok");
    7. });
    8. }
    9. } catch (Exception e) {
    10. e.printStackTrace();
    11. } finally {
    12. threadPool.shutdown(); //线程池使用完毕后需要关闭
    13. }

    部署后,规约扫描报错“不能使用Executors创建线程池”。

    4.【强制】 线程池不允许使用Executors去创建,而是通过ThreadPoolExecutor的方式,这样的处理方式让写的同学更加明确线程池的运行规则,规避资源耗尽的风险。

    说明:Executors返回的线程池对象的弊端如下:
    1)FixedThreadPool和SingleThreadPool:
    允许的请求队列长度为Integer.MAX_VALUE,可能会堆积大量的请求,从而导致OOM。
    2)CachedThreadPool
    允许的创建线程数量为Integer.MAX_VALUE,可能会创建大量的线程,从而导致OOM。

    修改2:改用ThreadPoolExecutor创建线程池

    1. ThreadFactory namedThreadFactory = new ThreadFactoryBuilder().setNameFormat("demo-pool-%d").build();
    2. ThreadPoolExecutor threadPool = new ThreadPoolExecutor(
    3. 2,
    4. size,
    5. 3,
    6. TimeUnit.SECONDS,
    7. new ArrayBlockingQueue<>(3),
    8. namedThreadFactory,
    9. new ThreadPoolExecutor.DiscardOldestPolicy()
    10. );
    11. ListMap<String, String>>> futureTasks = new ArrayList<>();
    12. try {
    13. for (int i = 0; i < size; i++) {
    14. final int finalI = i+1;
    15. long realInterval = interval != null ? interval * finalI : 10;
    16. final FutureTask<Map<String, String>> task = new FutureTask<>(() ->
    17. selfDefineFunction());
    18. futureTasks.add(task);
    19. threadPool.submit(task);
    20. }
    21. } catch (Exception e) {
    22. e.printStackTrace();
    23. } finally {
    24. threadPool.shutdown(); //线程池使用完毕后需要关闭
    25. }
    26. futureTasks.forEach(task -> {
    27. try {
    28. Map<String, ShakeResultDTO> oneResult = task.get();
    29. matchResultMap.putAll(oneResult);
    30. } catch (InterruptedException | ExecutionException e) {
    31. e.printStackTrace();
    32. }
    33. });

    小结

    1、手动创建-->Executors-->ThreadPoolExecutor

    2、返回值submit、不带返回值execute

    参考

    多线程之使用线程池创建线程
    线程池浅析 ThreadPoolExecutor FutureTask​​​​​​​

  • 相关阅读:
    【代码分析】初学解惑C++:函数适配器
    弘玑Cyclone2022产品发布会:人机交互数智助手——智能助手CIRI
    dask读取sql数据:MySQL
    SprimgMVC增删改查·
    开源网络协议栈onps诞生记
    配置中心微服务(Spring Cloud Config)
    【数据结构】单链表的增删查改
    vue新手入门之使用vue框架搭建用户登录注册案例,手动搭建webpack+Vue项目(附源码,图文详解,亲测有效)
    leetcode 145.二叉树的后序遍历
    QT找不到ffmpeg链接库解决方法
  • 原文地址:https://blog.csdn.net/welcome_zhang/article/details/126360338