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

迅速了解JDK线程池以及Spring线程池

  对于经常创建和销毁,使用量特别大的资源,比如并发情况下的线程,对性能影响非常大。所以我们一般​提前创建好多个线程,放入线程池中,使用时直接获取,使用完放入池中。可以避免频繁创建销毁,实现重复利用。

好处:

  • 提高响应速度(减少创建新线程的时间)
  • 降低资源消耗(重复利用线程池中的线程,不需要每次都创建)
  • 便于线程管理
    • 线程池大小
    • 最大线程数

对于Java多线程不是很熟悉的可以先去一篇看懂Java多线程

JDK自带的线程池(常用的有两个)

  • ExecutorService (这是一个普通的线程池,能够创建普通的线程)
  • ScheduledExecutorService(这个线程池创建的线程,每隔一段时间执行一个任务,它可以执行那种间隔的任务)
public class ThreadPoolTest {
//    JDK普通线程池
    private ExecutorService executorService = Executors.newFixedThreadPool(5);

//    JDK可执行定时任务的线程池
    private ScheduledExecutorService scheduledExecutorService = Executors.newScheduledThreadPool(5);

	//在main方法中启动线程,如果这个线程不挂掉的话,这个main会等待它执行,不会立刻就结束。
    // 但是Test方法,Junit方法不一样,它启动的子线程和当前线程是并发的,test方法后面如果没有逻辑,它立刻就结束了。
    //解决办法:在test方法启动一个线程以后想等一会,等他执行完以后在结束,可以让主线程或者当前线程sleep一会,阻塞一会
    //又因为sleep老是抛异常,为了调起来方便一点,稍微封装一下。
    private void sleep(long m){
        try{
            Thread.sleep(m);
        }catch (InterruptedException e){
            e.printStackTrace();
        }
    }

    //JDK普通线程池
    @Test
    public void test(){
        Runnable task = new Runnable() {
            @Override
            public void run() {
                logger.debug("Hello ExecutorService");
            }
        };

        for (int i = 0; i < 10; i++) {
            executorService.submit(task);//每调用一次这个submit方法,就会分配一个线程来执行这个线程体。
        }
        sleep(10000);
    }

    //JDK定时任务线程池
    @Test
    public void test1(){
        Runnable task = new Runnable() {
            @Override
            public void run() {
                logger.debug("Hello scheduledExecutorService");
            }
        };

        //scheduleAtFixedRate以固定的频率去执行
        //scheduleWithFixedDelay以固定的一个延迟去执行
        scheduledExecutorService.scheduleAtFixedRate(task,10000,1000, TimeUnit.MILLISECONDS);

        sleep(30000);
    }
}

Spring 线程池

  • ThreadPoolTaskExecutor(一个普通的线程池,创建普通的线程)
  • ThreadPoolTaskScheduler(创建的线程可以执行定时任务。)
@SpringBootTest
public class ThreadPoolTest {
	 //spring框架已经初始化好了,并且放入容器中了。
    //spring普通线程池
    @Autowired
    private ThreadPoolTaskExecutor taskExecutor;

    //spring 可以执行定时任务的线程池。
    @Autowired
    private ThreadPoolTaskScheduler taskScheduler;

	private void sleep(long m){
	        try{
	            Thread.sleep(m);
	        }catch (InterruptedException e){
	            e.printStackTrace();
	        }
	    }
	//spring 普通线程池
    @Test
    public void test2(){
        Runnable task = new Runnable() {
            @Override
            public void run() {
                logger.debug("Hello ThreadPoolTaskExecutor");
            }
        };

        for (int i = 0; i < 10; i++) {
            taskExecutor.submit(task);
        }
        sleep(10000);
    }

    //spring 定时任务线程池
    @Test
    public void test3(){
        Runnable task = new Runnable() {
            @Override
            public void run() {
                logger.debug("Hello ThreadPoolTaskScheduler");
            }
        };
        Date startTime = new Date(System.currentTimeMillis() + 10000);
        taskScheduler.scheduleAtFixedRate(task, startTime, 1000);//默认以毫秒为单位
        sleep(30000);
    }
}

在application.properties中的配置如下:

# TaskExecutionProperties(普通线程池)
# 核心线程数
spring.task.execution.pool.core-size=5
# 最大线程数
spring.task.execution.pool.max-size=15
# 队列的容量,如果15个还不够,就把任务先放在队列里,等有空闲线程了再分配。
spring.task.execution.pool.queue-capacity=100

# TaskSchedulingProperties(可执行定时任务的线程池)
spring.task.scheduling.pool.size=5

相关文章:

  • 前缀和与查分(一维前缀和,二维前缀和(子矩阵的和)一维差分、二维差分(差分矩阵))
  • 2022年是SEO行业凋谢的一年
  • CDR插件开发之Addon插件006 - 初体验:通过C#代码用外挂方式操作CDR中的对象
  • 【2020.09.01】 新学期,新气象
  • 基于云计算与深度学习的常见作物害虫识别系统的设计与实现
  • Flask 学习-22.可插拨视图MethodView类
  • 微信公众号如何获取查题搜题功能接口
  • 聚醋酸乙烯酯接枝聚苯乙烯PVAc-g-PSt微球/接枝-聚甲基丙烯酸甲酯表面(PS-acyl-Cl)的研究
  • 百度网盘的音乐怎么分享到qq音乐里?
  • Q_PLUGIN_METADATA
  • 【Java初阶】面向对象三大特性之继承
  • 标签传播算法(LPA)
  • ElasticSearch(版本7.8.1)中类型Long精度缺失
  • “两利四率” 、“两增一控三提高” 央企考核指标解读
  • Docker实战-部署GPE微服务的监控体系(二)
  • 30秒的PHP代码片段(1)数组 - Array
  • Gradle 5.0 正式版发布
  • PhantomJS 安装
  • php面试题 汇集2
  • 从伪并行的 Python 多线程说起
  • 和 || 运算
  • 基于HAProxy的高性能缓存服务器nuster
  • 快速构建spring-cloud+sleuth+rabbit+ zipkin+es+kibana+grafana日志跟踪平台
  • 类orAPI - 收藏集 - 掘金
  • 聊聊redis的数据结构的应用
  • 使用Tinker来调试Laravel应用程序的数据以及使用Tinker一些总结
  • 手写双向链表LinkedList的几个常用功能
  • 消息队列系列二(IOT中消息队列的应用)
  • 06-01 点餐小程序前台界面搭建
  • [Shell 脚本] 备份网站文件至OSS服务(纯shell脚本无sdk) ...
  • 新海诚画集[秒速5センチメートル:樱花抄·春]
  • 专访Pony.ai 楼天城:自动驾驶已经走过了“从0到1”,“规模”是行业的分水岭| 自动驾驶这十年 ...
  • ​Base64转换成图片,android studio build乱码,找不到okio.ByteString接腾讯人脸识别
  • ![CDATA[ ]] 是什么东东
  • # 手柄编程_北通阿修罗3动手评:一款兼具功能、操控性的电竞手柄
  • #宝哥教你#查看jquery绑定的事件函数
  • (delphi11最新学习资料) Object Pascal 学习笔记---第2章第五节(日期和时间)
  • (ZT)薛涌:谈贫说富
  • (安卓)跳转应用市场APP详情页的方式
  • (一)kafka实战——kafka源码编译启动
  • .NET CF命令行调试器MDbg入门(四) Attaching to Processes
  • .NET 的程序集加载上下文
  • .net 生成二级域名
  • .Net6 Api Swagger配置
  • .NET开发不可不知、不可不用的辅助类(三)(报表导出---终结版)
  • .NET命令行(CLI)常用命令
  • .NET是什么
  • .Net下的签名与混淆
  • /var/spool/postfix/maildrop 下有大量文件
  • @Autowired 与@Resource的区别
  • @staticmethod和@classmethod的作用与区别
  • [ Linux ] Linux信号概述 信号的产生
  • [4.9福建四校联考]
  • [ACTF2020 新生赛]Include
  • [android] 切换界面的通用处理