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


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

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

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

    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​​​​​​​

  • 相关阅读:
    ReactNative安卓首屏白屏优化
    七天学会C语言-第五天(函数)
    swiper动态渲染视图,解决答题问题
    Vuex的使用
    【0146】判断System V shared memory以前的段是否存在并正在使用?(3)
    openresty+etcd配置实现原理
    肺部阴影识别检测 matlab算法技术
    【图自动编码器】基础介绍 及 基于PyTorch的图自动编码器实例代码 | MLP应用于节点级别和图级别的任务实例(附实例代码+数据集)
    发明专利加急程序
    ts使用记录
  • 原文地址:https://blog.csdn.net/welcome_zhang/article/details/126360338