【JUC】006-线程池

叁歲伎倆 2023-10-05 14:19 189阅读 0赞

#

目录

一、概述

1、池化技术

2、线程池的好处

3、线程池知识核心

二、3个方法

1、Executors.newSingleThreadExecutor()

概述:

代码实现:

运行结果:

PS:

2、Executors.newFixedThreadPool(int num)

概述:

代码实现:

运行结果:

PS:

3、Executors.newCachedThreadPool()

概述:

代码实现:

运行结果:

PS:

三、7个参数

1、7个参数引出

构造函数及7个参数的解释:

阿里巴巴Java编程规约:

图解:

2、手动创建线程池

代码实现:

运行结果:

四、4种拒绝策略

1、不处理,抛出异常

2、哪来的回到哪里去

3、不处理,也不抛出异常

4、尝试获取最早被使用的线程资源

五、最大线程到底该如何设置

1、CPU密集型和IO密集型

CPU密集型:

IO密集型:


一、概述

1、池化技术

程序的运行需要占用系统资源,为了减少系统消耗、提升系统性能并方便管理,就有了池化技术;

2、线程池的好处

线程复用、可以控制最大并发数、管理线程;

3、线程池知识核心

3个方法、7个参数、4种拒绝策略;

二、3个方法

1、Executors.newSingleThreadExecutor()

概述:

创建单个线程的线程池;

代码实现:

  1. package com.zibo.pool;
  2. import java.util.concurrent.ExecutorService;
  3. import java.util.concurrent.Executors;
  4. //Executors工具类:三个方法
  5. //使用线程池创建线程
  6. public class Demo01 {
  7. public static void main(String[] args) {
  8. ExecutorService threadPool = Executors.newSingleThreadExecutor();//创建单个线程的线程池
  9. //Executors.newFixedThreadPool(5);//创建固定线程数量的线程池
  10. //Executors.newCachedThreadPool();//创建不固定线程数量的线程池,大小自动伸缩
  11. try {
  12. for (int i = 0; i < 10; i++) {
  13. //创建线程并执行
  14. threadPool.execute(()->{
  15. System.out.println(Thread.currentThread().getName() + "==>ok!");
  16. });
  17. }
  18. } catch (Exception e) {
  19. e.printStackTrace();
  20. } finally {
  21. //线程池用完需要关闭,释放内存
  22. threadPool.shutdown();
  23. }
  24. }
  25. }

运行结果:

  1. pool-1-thread-1==>ok!
  2. pool-1-thread-1==>ok!
  3. pool-1-thread-1==>ok!
  4. pool-1-thread-1==>ok!
  5. pool-1-thread-1==>ok!
  6. pool-1-thread-1==>ok!
  7. pool-1-thread-1==>ok!
  8. pool-1-thread-1==>ok!
  9. pool-1-thread-1==>ok!
  10. pool-1-thread-1==>ok!

PS:

因为线程池只有一个线程,所以线程名字都一样;

2、Executors.newFixedThreadPool(int num)

概述:

创建固定线程数量的线程池;

代码实现:

  1. package com.zibo.pool;
  2. import java.util.concurrent.ExecutorService;
  3. import java.util.concurrent.Executors;
  4. //Executors工具类:三个方法
  5. //使用线程池创建线程
  6. public class Demo01 {
  7. public static void main(String[] args) {
  8. //ExecutorService threadPool = Executors.newSingleThreadExecutor();//创建单个线程的线程池
  9. ExecutorService threadPool = Executors.newFixedThreadPool(5);//创建固定线程数量的线程池
  10. //Executors.newCachedThreadPool();//创建不固定线程数量的线程池,大小自动伸缩
  11. try {
  12. for (int i = 0; i < 10; i++) {
  13. //创建线程并执行
  14. threadPool.execute(()->{
  15. System.out.println(Thread.currentThread().getName() + "==>ok!");
  16. });
  17. }
  18. } catch (Exception e) {
  19. e.printStackTrace();
  20. } finally {
  21. //线程池用完需要关闭,释放内存
  22. threadPool.shutdown();
  23. }
  24. }
  25. }

运行结果:

  1. pool-1-thread-1==>ok!
  2. pool-1-thread-4==>ok!
  3. pool-1-thread-3==>ok!
  4. pool-1-thread-2==>ok!
  5. pool-1-thread-3==>ok!
  6. pool-1-thread-4==>ok!
  7. pool-1-thread-5==>ok!
  8. pool-1-thread-1==>ok!
  9. pool-1-thread-3==>ok!
  10. pool-1-thread-2==>ok!

PS:

线程池里面有5条线程,所以会出现不同的线程被调用;

3、Executors.newCachedThreadPool()

概述:

创建不固定线程数量的线程池,大小自动伸缩;

代码实现:

  1. package com.zibo.pool;
  2. import java.util.concurrent.ExecutorService;
  3. import java.util.concurrent.Executors;
  4. //Executors工具类:三个方法
  5. //使用线程池创建线程
  6. public class Demo01 {
  7. public static void main(String[] args) {
  8. //ExecutorService threadPool = Executors.newSingleThreadExecutor();//创建单个线程的线程池
  9. //ExecutorService threadPool = Executors.newFixedThreadPool(5);//创建固定线程数量的线程池
  10. ExecutorService threadPool = Executors.newCachedThreadPool();//创建不固定线程数量的线程池,大小自动伸缩
  11. try {
  12. for (int i = 0; i < 10; i++) {
  13. //创建线程并执行
  14. threadPool.execute(()->{
  15. System.out.println(Thread.currentThread().getName() + "==>ok!");
  16. });
  17. }
  18. } catch (Exception e) {
  19. e.printStackTrace();
  20. } finally {
  21. //线程池用完需要关闭,释放内存
  22. threadPool.shutdown();
  23. }
  24. }
  25. }

运行结果:

  1. pool-1-thread-1==>ok!
  2. pool-1-thread-5==>ok!
  3. pool-1-thread-4==>ok!
  4. pool-1-thread-3==>ok!
  5. pool-1-thread-2==>ok!
  6. pool-1-thread-8==>ok!
  7. pool-1-thread-9==>ok!
  8. pool-1-thread-7==>ok!
  9. pool-1-thread-6==>ok!
  10. pool-1-thread-5==>ok!

PS:

自由伸缩,可大可小;

三、7个参数

1、7个参数引出

通过源码我们发现,上面三个创建线程池的方法都是使用ThreadPoolExecutor类的构造函数创建的,而且ThreadPoolExecutor类的构造函数里面有7个参数;

构造函数及7个参数的解释:

  1. public ThreadPoolExecutor(int corePoolSize, //核心线程池大小
  2. int maximumPoolSize, //最大线程池大小
  3. long keepAliveTime, //保持存活时间(超时了没有被使用就会被释放)
  4. TimeUnit unit, //超时的时间单位
  5. BlockingQueue<Runnable> workQueue, //阻塞队列
  6. ThreadFactory threadFactory, //线程工厂,用于创建线程,一般不变
  7. RejectedExecutionHandler handler) {//拒绝策略
  8. if (corePoolSize < 0 ||
  9. maximumPoolSize <= 0 ||
  10. maximumPoolSize < corePoolSize ||
  11. keepAliveTime < 0)
  12. throw new IllegalArgumentException();
  13. if (workQueue == null || threadFactory == null || handler == null)
  14. throw new NullPointerException();
  15. this.acc = System.getSecurityManager() == null ?
  16. null :
  17. AccessController.getContext();
  18. this.corePoolSize = corePoolSize;
  19. this.maximumPoolSize = maximumPoolSize;
  20. this.workQueue = workQueue;
  21. this.keepAliveTime = unit.toNanos(keepAliveTime);
  22. this.threadFactory = threadFactory;
  23. this.handler = handler;
  24. }

阿里巴巴Java编程规约:

watermark_type_ZmFuZ3poZW5naGVpdGk_shadow_10_text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3FxXzI5Njg5MzQz_size_16_color_FFFFFF_t_70

图解:

watermark_type_ZmFuZ3poZW5naGVpdGk_shadow_10_text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3FxXzI5Njg5MzQz_size_16_color_FFFFFF_t_70 1

2、手动创建线程池

代码实现:

  1. package com.zibo.pool;
  2. import java.util.concurrent.*;
  3. //Executors工具类:三个方法
  4. //使用线程池创建线程
  5. public class Demo01 {
  6. public static void main(String[] args) {
  7. //自定义线程池,在工作中只会用这种方式
  8. ExecutorService threadPool = new ThreadPoolExecutor(
  9. 2,//核心线程数量
  10. 5,//最大线程数量
  11. 3,//保持存活时间
  12. TimeUnit.SECONDS,//时间单位
  13. new LinkedBlockingDeque<>(3),//阻塞队列
  14. Executors.defaultThreadFactory(),//线程工厂
  15. new ThreadPoolExecutor.AbortPolicy()//所有线程都被占用了,阻塞队列也满了,还有线程进来,就不再处理,并抛出异常
  16. );
  17. try {
  18. //最大承载量 = 最大线程数量 + 阻塞队列数量;
  19. for (int i = 0; i < 10; i++) {//这里最大是8,超出就会抛出异常
  20. //创建线程并执行
  21. threadPool.execute(()->{
  22. System.out.println(Thread.currentThread().getName() + "==>ok!");
  23. });
  24. }
  25. } catch (Exception e) {
  26. e.printStackTrace();
  27. } finally {
  28. //线程池用完需要关闭,释放内存
  29. threadPool.shutdown();
  30. }
  31. }
  32. }

运行结果:

watermark_type_ZmFuZ3poZW5naGVpdGk_shadow_10_text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3FxXzI5Njg5MzQz_size_16_color_FFFFFF_t_70 2

四、4种拒绝策略

1、不处理,抛出异常

new ThreadPoolExecutor.AbortPolicy():所有线程都被占用了,阻塞队列也满了,还有线程进来,就不再处理,并抛出异常;

2、哪来的回到哪里去

new ThreadPoolExecutor.CallerRunsPolicy()

watermark_type_ZmFuZ3poZW5naGVpdGk_shadow_10_text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3FxXzI5Njg5MzQz_size_16_color_FFFFFF_t_70 3

3、不处理,也不抛出异常

new ThreadPoolExecutor.DiscardPolicy()

watermark_type_ZmFuZ3poZW5naGVpdGk_shadow_10_text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3FxXzI5Njg5MzQz_size_16_color_FFFFFF_t_70 4

4、尝试获取最早被使用的线程资源

new ThreadPoolExecutor.DiscardOldestPolicy()

watermark_type_ZmFuZ3poZW5naGVpdGk_shadow_10_text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3FxXzI5Njg5MzQz_size_16_color_FFFFFF_t_70 5

五、最大线程到底该如何设置

1、CPU密集型和IO密集型

CPU密集型:

cpu是多少核,就是几,可以保持CPU效率最高;

  1. //获取核数:
  2. Runtime.getRuntime().availableProcessors()

IO密集型:

判断程序中耗IO的线程数量,设置大于此数量(一般两倍)即可;

发表评论

表情:
评论列表 (有 0 条评论,189人围观)

还没有评论,来说两句吧...

相关阅读

    相关 JUC-线

    ThreadPoolExecutor 使用 int 的高 3 位来表示线程池状态,低 29 位表示线程数量。这些信息存储在原子变量ctl中,将线程状态与线程数合二为一,这...

    相关 JUC包(五) 线

    前言 在前面的篇章中.我们将解了`线程/锁/多线程容器`.本章我们将介绍一个用于管理线程的容器:`线程池`. -------------------- 正文 线