Java多线程之线程池(合理分配资源)

痛定思痛。 2023-09-27 09:57 118阅读 0赞

前言必读

读者手册(必读)_云边的快乐猫的博客-CSDN博客

一、故事讲解

1.故事

有一家月饼店开业了,店里面有3个核心员工,这些人负责月饼的制作,3人同时工作制作100个月饼,每个人做完自己的月饼就会忙着下一个月饼的制作。因为总不可能要100个员工来做这些月饼吧,那也太浪费了。这些月饼的制作顺序是按照先后排队等待被制作的。

中秋佳节来临了,这些月饼的单子一下子赶了起来。忙不过来那就只能请一些临时员工来帮忙,但是核心员工忙得过来就用不上临时员工了。同时根据这些核心员工和临时员工的工作能力来判断接单情况,超过工作情况就不接单了,等先忙完这些再说。忙完这些月饼的制作,临时员工就会被辞退的。

2.怎么判断忙不过来,该找临时员工了?

3个核心线程都在忙,100个任务都排满了,那就只能找临时员工了

3.详解:

月饼店—-线程池

3个核心员工—3个核心线程

100个月饼—-100个任务

临时员工—-临时线程

辞退—-销毁临时线程(自动销毁,不用手动去设置)

二、概念讲解

1.什么是线程池?

答:一个可以复用线程的技术

2.如果不用线程池会怎么样?

答: 如果用户每发起一个请求,后台就会创建一个新的线程来进行处理。而一直开新线程的开销是很大的,这样会严重影响系统的性能。

3.怎么创建线程池?

答:创建ExecutorService接口的实现类,创建方式有两种

4.创建线程过多导致的问题?

答: 创建线程过多会占用CPU资源的,导致卡顿

三、两种线程池的创建方式

方式一:使用ExecutorService的实现类ThreadPoolExecutor去自己创建一个线程池对象。

(1)处理Runnable任务

步骤:

1.创建Runnable线程类

2.在main方法里面创建ExecutorService pool = new ThreadPoolExecutor();-—括号里面填上线程的定义规则

3.创建Runnable target = new MyRunnable();

4.创建线程pool.execute(target);

代码例子:

  1. package bao;
  2. import java.util.concurrent.*;
  3. public class Test {
  4. public static void main(String[] args) {
  5. /* //一、1.自己定义一个线程池对象----多态的写法
  6. * 3个核心员工(3个核心线程)
  7. * 5个总员工,那就是临时工是2个(总线程5个)
  8. * 6s,TimeUnit.SECONDS----临时工没事干就被销毁辞退了
  9. * 员工最多只能5个
  10. * Executors.defaultThreadFactory()-----默认线程工厂
  11. *new ThreadPoolExecutor.AbortPolicy()-----拒绝别人
  12. * */
  13. ExecutorService pool = new ThreadPoolExecutor(3,5,6, TimeUnit.SECONDS,new ArrayBlockingQueue<>(5), Executors.defaultThreadFactory(),new ThreadPoolExecutor.AbortPolicy());
  14. //5.new创建线程
  15. Runnable target = new MyRunnable();
  16. //线程多少来自于上面自定义
  17. //因为线程方法都一样。调用有可能一下子就调用线程4-3-1了,那剩余的有可能会分配到任务,有可能闲着
  18. pool.execute(target);//核心线程1
  19. pool.execute(target);//核心线程2
  20. pool.execute(target);//核心线程3
  21. //创建临时线程(线程名字都一样,5个不一定哪个是核心,哪个是临时,分辨不出来)
  22. pool.execute(target);//线程4
  23. pool.execute(target);//线程5
  24. //为什么一些超过5条线程还能运行不报错呢?比如线程池里面只有5个位置,但是看着这么多线程都在,以为可以调用超过5个,一旦调用超过了5个就会报错
  25. //关闭线程(几乎不会用到,了解了就好了)
  26. //pool.shutdownNow();立即强制关掉,即使任务没有完成也会被关掉,会丢失任务
  27. //pool.shutdown();等待任务完成后再关闭,相对柔和
  28. }
  29. }
  30. //二、创建线程对象类
  31. //2.创建线程类
  32. class MyRunnable implements Runnable{
  33. //3.重写run方法
  34. @Override
  35. public void run() {
  36. //4.定义线程要执行的任务
  37. for (int i = 0; i < 5; i++) {
  38. System.out.println(Thread.currentThread().getName()+"输出了:月饼"+i);//Thread.currentThread().getName()可以知道当前哪个线程处理任务
  39. }
  40. }
  41. }

运行结果:

pool-1-thread-3输出了:月饼0
pool-1-thread-1输出了:月饼0
pool-1-thread-2输出了:月饼0
pool-1-thread-1输出了:月饼1
pool-1-thread-3输出了:月饼1
pool-1-thread-1输出了:月饼2
pool-1-thread-2输出了:月饼1
pool-1-thread-1输出了:月饼3
pool-1-thread-3输出了:月饼2
pool-1-thread-1输出了:月饼4
pool-1-thread-2输出了:月饼2
pool-1-thread-3输出了:月饼3
pool-1-thread-2输出了:月饼3
pool-1-thread-3输出了:月饼4
pool-1-thread-1输出了:月饼0
pool-1-thread-3输出了:月饼0
pool-1-thread-2输出了:月饼4
pool-1-thread-3输出了:月饼1
pool-1-thread-1输出了:月饼1
pool-1-thread-3输出了:月饼2
pool-1-thread-1输出了:月饼2
pool-1-thread-3输出了:月饼3
pool-1-thread-1输出了:月饼3
pool-1-thread-3输出了:月饼4
pool-1-thread-1输出了:月饼4

(2)处理Callable任务

步骤:

1.创建Callable线程类

2.在main方法里面创建ExecutorService pool = new ThreadPoolExecutor();-—括号里面填上线程的定义规则

  1. 创建线程Future f1 = pool.submit(new MyCallable(这里输入传入的值));

4.输出结果System.out.println(f1.get());//有异常直接抛出就好了

代码例子:

  1. package bao;
  2. import java.util.concurrent.*;
  3. public class Test {
  4. public static void main(String[] args) throws ExecutionException, InterruptedException {
  5. //二、4.自定义线程的规则(详情看上一题)
  6. ExecutorService pool = new ThreadPoolExecutor(3,5,6, TimeUnit.SECONDS,new ArrayBlockingQueue<>(5),Executors.defaultThreadFactory(),new ThreadPoolExecutor.AbortPolicy());
  7. //5.把任务给线程池
  8. Future<String> f1 = pool.submit(new MyCallable(100));
  9. Future<String> f2 = pool.submit(new MyCallable(200));
  10. Future<String> f3 = pool.submit(new MyCallable(300));
  11. Future<String> f4 = pool.submit(new MyCallable(400));
  12. Future<String> f5 = pool.submit(new MyCallable(500));
  13. //6.输出结果
  14. System.out.println(f1.get());//有异常直接抛出就好了
  15. System.out.println(f2.get());
  16. System.out.println(f3.get());
  17. System.out.println(f4.get());
  18. System.out.println(f5.get());
  19. }
  20. }
  21. //一、1.创建线程任务类Callable
  22. class MyCallable implements Callable<String>{
  23. private int n;
  24. public MyCallable(int n){
  25. this.n=n;
  26. }
  27. //2.重写Call方法
  28. @Override
  29. public String call() throws Exception {
  30. //3.写线程要执行的具体任务代码
  31. int sum = 0;
  32. for (int i = 0; i < n; i++) {
  33. sum+=i;
  34. }
  35. return Thread.currentThread().getName()+"执行1-"+n+"的和是"+sum;
  36. }
  37. }

运行结果:

pool-1-thread-1执行1-100的和是4950
pool-1-thread-2执行1-200的和是19900
pool-1-thread-3执行1-300的和是44850
pool-1-thread-2执行1-400的和是79800
pool-1-thread-3执行1-500的和是124750

方式二:使用Executors(线程池的工具类)调用方法返回不同特点的线程池对象。(不适合做大型互联网场景的线程池方案,阿里巴巴手册不允许使用)

这个方式使用的例子方式就是固定了线程的数量使用。

步骤:

1.创建线程类

2.在main方法里面创建ExecutorService pool = Executors.newFixedThreadPool(这里固定线程数量);

3.创建线程pool.execute(new MyRunnable());

代码例子:

  1. package bao;
  2. import java.util.concurrent.ExecutorService;
  3. import java.util.concurrent.Executors;
  4. public class Test {
  5. public static void main(String[] args) {
  6. //二、5.创建固定线程池数据的线程池
  7. ExecutorService pool = Executors.newFixedThreadPool(3);//这里固定线程池里面只能有3个线程
  8. //6.创建线程
  9. pool.execute(new MyRunnable());//线程1
  10. pool.execute(new MyRunnable());//线程2
  11. pool.execute(new MyRunnable());//线程3
  12. //创建多余的线程也不要了,看都不看,就算上面的线程睡眠了
  13. }
  14. }
  15. //一、1.创建线程对象类
  16. //2.创建线程类
  17. class MyRunnable implements Runnable{
  18. //3.重写run方法
  19. @Override
  20. public void run() {
  21. //4.定义线程要执行的任务
  22. for (int i = 0; i < 5; i++) {
  23. System.out.println(Thread.currentThread().getName()+"输出了:月饼"+i);//Thread.currentThread().getName()可以知道当前哪个线程处理任务
  24. }
  25. }
  26. }

运行结果:

pool-1-thread-3输出了:月饼0
pool-1-thread-2输出了:月饼0
pool-1-thread-1输出了:月饼0
pool-1-thread-2输出了:月饼1
pool-1-thread-3输出了:月饼1
pool-1-thread-2输出了:月饼2
pool-1-thread-1输出了:月饼1
pool-1-thread-2输出了:月饼3
pool-1-thread-3输出了:月饼2
pool-1-thread-2输出了:月饼4
pool-1-thread-1输出了:月饼2
pool-1-thread-3输出了:月饼3
pool-1-thread-1输出了:月饼3
pool-1-thread-3输出了:月饼4
pool-1-thread-1输出了:月饼4

旁白:线程池的出现就是为了避免资源的浪费,合理分配线程的资源,创建方式有两种,主要用第一种,第二种工具线程不建议使用。

发表评论

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

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

相关阅读

    相关 java线线

    java多线程之线程池 池化技术 程序的运行,其本质上,是对系统资源(CPU、内存、磁盘、网络等等)的使用。如何高效的使用这些资源是我们编程优化演进的一个方向。今天说

    相关 java线线(ExecutorService)

           创建新线程的服务器在创建和销毁线程上花费的时间和消耗的系统资源要比花在处理实际的用户请求的时间和资源更多,线程池为线程生命周期开销问题和资源不足问题提供了解决方案

    相关 线线

    前言: 1. 系统启动一个新线程的成本是比较高的,因为它涉及与操作系统交互。在这种情形下,使用线程池可以很好地提高性能,尤其是当程序中需要创建大量生存期很短暂的线程时,