• Java中常用的四种线程池


            在Java中使用线程池,可以用ThreadPoolExecutor的构造函数直接创建出线程池实例,在Executors类中,为我们提供了常用线程池的创建方法。

    接下来我们就来了解常用的四种:

    newFixedThreadPool

            首先,看一下这种线程池的创建方法:

    1. public static ExecutorService newFixedThreadPool(int nThreads) {
    2. return new ThreadPoolExecutor(nThreads, nThreads,
    3. 0L, TimeUnit.MILLISECONDS,
    4. new LinkedBlockingQueue<Runnable>());
    5. }

            从构造方法可以看出,它创建了一个固定大小的线程池,每次提交一个任务就创建一个线程,直到线程达到线程池的最大值nThreads。线程池的大小一旦达到最大值后,再有新的任务提交时则放入无界阻塞队列中,等到有线程空闲时,再从队列中取出任务继续执行。 那么,如何使用newFixedThreadPool呢?我们来举个例子:

    1. public class OneMoreStudy {
    2. public static void main(String[] args) {
    3. ExecutorService fixedThreadPool = Executors.newFixedThreadPool(3);
    4. for (int i = 0; i < 5; i++) {
    5. final int index = i;
    6. fixedThreadPool.execute(new Runnable() {
    7. @Override
    8. public void run() {
    9. try {
    10. SimpleDateFormat sdf = new SimpleDateFormat(
    11. "HH:mm:ss");
    12. System.out.println("运行时间: " +
    13. sdf.format(new Date()) + " " + index);
    14. Thread.sleep(2000);
    15. } catch (InterruptedException e) {
    16. e.printStackTrace();
    17. }
    18. }
    19. });
    20. }
    21. fixedThreadPool.shutdown();
    22. }
    23. }

            上面的例子中创建了一个固定大小为3的线程池,然后在线程池提交了5个任务。在提交第4个任务时,因为线程池的大小已经达到了3并且前3个任务在运行中,所以第4个任务被放入了队列,等待有空闲的线程时再被运行。运行结果如下(注意前3个任务和后2个任务的运行时间):

    1. 运行时间: 08:09:02 1
    2. 运行时间: 08:09:02 2
    3. 运行时间: 08:09:02 0
    4. 运行时间: 08:09:04 4
    5. 运行时间: 08:09:04 3

    newCachedThreadPool

            首先,看一下这种线程池的创建方法:

    1. public static ExecutorService newCachedThreadPool() {
    2. return new ThreadPoolExecutor(0, Integer.MAX_VALUE,
    3. 60L, TimeUnit.SECONDS,
    4. new SynchronousQueue<Runnable>());
    5. }

            从构造方法可以看出,它创建了一个可缓存的线程池。当有新的任务提交时,有空闲线程则直接处理任务,没有空闲线程则创建新的线程处理任务,队列中不储存任务。线程池不对线程池大小做限制,线程池大小完全依赖于操作系统(或者说JVM)能够创建的最大线程大小。如果线程空闲时间超过了60秒就会被回收。 那么,如何使用newCachedThreadPool呢?我们来举个例子:

    1. public class OneMoreStudy {
    2. public static void main(String[] args) {
    3. ExecutorService cachedThreadPool = Executors.newCachedThreadPool();
    4. for (int i = 0; i < 5; i++) {
    5. final int index = i;
    6. cachedThreadPool.execute(new Runnable() {
    7. @Override
    8. public void run() {
    9. try {
    10. SimpleDateFormat sdf = new SimpleDateFormat(
    11. "HH:mm:ss");
    12. System.out.println("运行时间: " +
    13. sdf.format(new Date()) + " " + index);
    14. Thread.sleep(2000);
    15. } catch (InterruptedException e) {
    16. e.printStackTrace();
    17. }
    18. }
    19. });
    20. }
    21. cachedThreadPool.shutdown();
    22. }
    23. }

            因为这种线程有新的任务提交,就会创建新的线程(线程池中没有空闲线程时),不需要等待,所以提交的5个任务的运行时间是一样的,运行结果如下:

    1. 运行时间: 08:45:18 2
    2. 运行时间: 08:45:18 1
    3. 运行时间: 08:45:18 3
    4. 运行时间: 08:45:18 4
    5. 运行时间: 08:45:18 0

    newSingleThreadExecutor

            首先,看一下这种线程池的创建方法:

    1. public static ExecutorService newSingleThreadExecutor() {
    2. return new FinalizableDelegatedExecutorService
    3. (new ThreadPoolExecutor(1, 1,
    4. 0L, TimeUnit.MILLISECONDS,
    5. new LinkedBlockingQueue<Runnable>()));
    6. }

            从构造方法可以看出,它创建了一个单线程化的线程池,它只会用唯一的工作线程来执行任务,保证所有任务按照指定顺序执行。 那么,如何使用newSingleThreadExecutor呢?我们来举个例子:

    1. public class OneMoreStudy {
    2. public static void main(String[] args) {
    3. ExecutorService singleThreadExecutor = Executors.newSingleThreadExecutor();
    4. for (int i = 0; i < 5; i++) {
    5. final int index = i;
    6. singleThreadExecutor.execute(new Runnable() {
    7. @Override
    8. public void run() {
    9. try {
    10. SimpleDateFormat sdf = new SimpleDateFormat(
    11. "HH:mm:ss");
    12. System.out.println("运行时间: " +
    13. sdf.format(new Date()) + " " + index);
    14. Thread.sleep(2000);
    15. } catch (InterruptedException e) {
    16. e.printStackTrace();
    17. }
    18. }
    19. });
    20. }
    21. singleThreadExecutor.shutdown();
    22. }
    23. }

            因为该线程池类似于单线程执行,所以先执行完前一个任务后,再顺序执行下一个任务, 运行结果如下:

    1. 运行时间: 08:54:17 0
    2. 运行时间: 08:54:19 1
    3. 运行时间: 08:54:21 2
    4. 运行时间: 08:54:23 3
    5. 运行时间: 08:54:25 4

            有的同学可能会质疑:既然类似于单线程执行,那么这种线程池还有存在的必要吗?这里的单线程执行指的是线程池内部,从线程池外的角度看,主线程在提交任务到线程池时并没有阻塞,仍然是异步的。

    newScheduledThreadPool

            这个方法创建了一个固定大小的线程池,支持定时及周期性任务执行。 首先看一下定时执行的例子:

    1. public class OneMoreStudy {
    2. public static void main(String[] args) {
    3. final SimpleDateFormat sdf = new SimpleDateFormat("HH:mm:ss");
    4. ScheduledExecutorService scheduledThreadPool = Executors.newScheduledThreadPool(3);
    5. System.out.println("提交时间: " + sdf.format(new Date()));
    6. scheduledThreadPool.schedule(new Runnable() {
    7. @Override
    8. public void run() {
    9. System.out.println("运行时间: " + sdf.format(new Date()));
    10. }
    11. }, 3, TimeUnit.SECONDS);
    12. scheduledThreadPool.shutdown();
    13. }
    14. }

            使用该线程池的schedule方法,延迟3秒钟后执行任务,运行结果如下:

    1. 提交时间: 09:11:39
    2. 运行时间: 09:11:42

            再看一下周期执行的例子:

    1. public class OneMoreStudy {
    2. public static void main(String[] args) {
    3. final SimpleDateFormat sdf = new SimpleDateFormat("HH:mm:ss");
    4. ScheduledExecutorService scheduledThreadPool = Executors.newScheduledThreadPool(3);
    5. System.out.println("提交时间: " + sdf.format(new Date()));
    6. scheduledThreadPool.scheduleAtFixedRate(new Runnable() {
    7. @Override
    8. public void run() {
    9. System.out.println("运行时间: " + sdf.format(new Date()));
    10. }
    11. }, 1, 3, TimeUnit.SECONDS);
    12. Thread.sleep(10000);
    13. scheduledThreadPool.shutdown();
    14. }
    15. }

            使用该线程池的scheduleAtFixedRate方法,延迟1秒钟后每隔3秒执行一次任务,运行结果如下:

    1. 提交时间: 09:23:20
    2. 运行时间: 09:23:21
    3. 运行时间: 09:23:24
    4. 运行时间: 09:23:27

    Java学习视频

    Java基础:

    Java300集,Java必备优质视频_手把手图解学习Java,让学习成为一种享受

    Java项目:

    【Java游戏项目】1小时教你用Java语言做经典扫雷游戏_手把手教你开发游戏

    【Java毕业设计】OA办公系统项目实战_OA员工管理系统项目_java开发

     

  • 相关阅读:
    SpringMvc项目部署
    ORACLE连接不上 Linux网络 端口 问题判断
    【附源码】计算机毕业设计java租车信息管理系统设计与实现
    go泛型使用方法
    client-go实战之八:更新资源时的冲突错误处理
    【Python基础】常用模块学习:sys|os|pytest
    程序环境和预处理
    【leetcode】剑指 Offer II 007. 数组中和为 0 的三个数(双指针)
    2020java面试总结
    前端页面左右布局,点击左div增加动画过渡效果
  • 原文地址:https://blog.csdn.net/java_0000/article/details/125432514