库调多了,都忘了最基础的概念 -《线程池篇》
🍁 作者:知识浅谈,CSDN 博客专家,阿里云签约博主,InfoQ 签约博主,华为云云享专家
📌 擅长领域:全栈工程师、爬虫、ACM 算法
💒 公众号:知识浅谈
XXXX 总结🤞这次都给他拿下🤞
正菜来了⛳⛳⛳
🎈为什么需要线程池?什么是池化技术?
线程池用于管理线程的创建与销毁。
相比于单线程的频繁创建与销毁消耗的资源,由线程池来管理创建与销毁节省了频繁创建与销毁的时间。
其次便于对所有的线程进行管理。
线程的重复使用,提高了利用率
节省了创建线程的时间,需要线程的时候可以快速使用。池化技术:就是提前准备好一些资源,在需要的时候就可以直接使用这些资源,并且能重复使用。如数据库连接池,线程池这些。
🎈线程池有几种创建方式?推荐使用哪种?
自动创建提供的有四种:
newFixedThreadPool:创建固定数量的线程池。核心线程数=最大线程数=固定的
newCachedThreadPool:能持续创建线程,因为最大的线程数量设置的为 Integer.MAX_VALUE
newSingleThreadExecutor:只能创建单个线程的线程池,核心线程数=最大线程数=1
newScheduledThreadPool:创建一个可以执行延迟任务的线程池。
手动创建 ThreadPoolExecutor 手动设置参数,一般都推荐根据业务自己设置不同的参数。
🎈说一下线程池 7 个参数的含义?
kernel-size:核心线程数也就是线程池中一直存活的线程数 maximumPoolSize :最大线程数,线程池中最多创建的线程数 keepAliveTime:指的是除了核心线程外的空闲线程存活时间。TimeUnit:存活时间的时间单位。BlockingQueue:线程池的任务队列。ThreadFactory:创建线程的工厂。RejectedExecutionHandler:拒绝策略。
🎈什么是守护线程?它和用户线程有什么区别?
守护线程是对一个线程进行监视的线程。守护线程是为用户线程服务的,当被守护线程死亡的时候,守护线程也就死亡了。需要注意的一点:守护线程的设置 setDaemon(true)也就是设置自己的线程为守护线程之后再启动 start()。
🎈为什么创建线程池一定要用 ThreadPoolExecutor?
之所以使用手动指定 ThreadPoolExecutor 的参数创建线程池,因为这种方式可以通过参数来控制最大任务数和拒绝策略,不仅能根据业务灵活变化,还能让线程池的执行更加透明和可控,降低资源消耗。
🎈线程池有哪些状态?状态是如何转换的?
Running:运行状态:线程池运行的时候
shutdown:关闭状态:不再接受其他的任务,把当前正在执行的任务和队列中的任务执行完。
stop:停滞状态:不再接受其他的任务,并且抛弃当前正在执行的任务和队列中的任务。
tiding:整理状态,所有的任务执行完之后,调用 terminated()方法
terminated:销毁状态,当执行完线程池的 terminated() 方法之后就会变为此状态。
🎈如何使用线程池执行定时任务?
正如上边说到的使用 ScheduledThreadPool 执行定时任务。具体的方法有如下三种:
只执行一次定时任务,使用 schedule 方法执行定时任务
执行多次定时任务,使用 scheduleAtFixedRate 方法执行定时任务
执行多次定时任务,使用 scheduleWithFixedDelay 方法执行定时任务
🎈如何判断线程池已经执行完所有任务了?
使用 isTerminated 方法判断。
使用 getCompletedTaskCount 方法判断。
使用 CountDownLatch 判断。这个在开始设置个数,每个线程执行完,调用 countDown()
使用 CyclicBarrier 判断。
🍚总结
以上就是关于线程池使用的总结,希望有所帮助。
版权声明: 本文为 InfoQ 作者【知识浅谈】的原创文章。
原文链接:【http://xie.infoq.cn/article/5d1394750bd829c424c323bd1】。
本文遵守【CC-BY 4.0】协议,转载请保留原文出处及本版权声明。
评论