多线程-(线程池)

1.线程池

定义:使用池化技术来管理线程和使用线程的方式。
线程池有两个重要的对象:

1.线程
2.工作队列(Integer最大值)

(1)线程的缺点

  • 1.线程的创建它会开辟本地方法栈、虚拟机栈、程序计数器成线程私有的内存,同时销毁的时候需要销毁这三个区域,因此频繁的创建和销毁会消耗系统资源。
  • 2.在任务量远远大于线程可以处理的任务量的时候,并不能友好拒绝任务。

(2)线程池的优点

  • 1.可以避免频繁的创建和销毁线程带来的系统性能的开销。
  • 2.可以更好的管理线程的个数和资源的个数。
  • 3.拥有更多的功能,比如线程池可以进行定时任务的执行。
  • 4.线程池可以更友好的拒绝不能处理的任务。

(3)线程池的创建

①创建固定个数的线程池

//创建一个固定个数的线程池
  ExecutorService executorService= Executors.newFixedThreadPool(10);
    //执行任务
     for (int i = 0; i <15 ; i++) {
        executorService.execute(new Runnable() {
             @Override
          public void run() {
             System.out.println("线程名:"+Thread.currentThread().getName());//拿到当前线程的名称
                }
            });
        }

②创建带缓存线程池

使用场景:短期有大量任务的时候可以使用newCachedThreadPool()

//创建带缓存的线程池
   ExecutorService executorService= Executors.newCachedThreadPool();
     for (int i = 0; i <10 ; i++) {
        executorService.execute(new Runnable() {
           @Override
           public void run() {
            System.out.println("线程名:"+Thread.currentThread().getName());
                }
            });
        }

③创建可以执行定时任务的线程池

//创建执行定时任务的线程池
 ScheduledExecutorService scheduledExecutorService= Executors.newScheduledThreadPool(2);
  System.out.println("设置定时任务之前"+new Date());
      // 执行定时任务
      //scheduleAtFixedRate:以上次任务的开始时间作为下次任务的开始时间
   scheduledExecutorService.scheduleAtFixedRate(new Runnable() {
        @Override
         public void run() {
        try {
            Thread.sleep(1000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
             System.out.println("执行任务"+new Date());
              }
         },1,3, TimeUnit.SECONDS);

参数1:线程执行的任务Runnable
参数2:延迟一段时间执行
参数3:定时任务执行的频率
参数4:配合参数2和参数3使用的,它规定了时间的单位
注意:
在这里插入图片描述

④创建单线程执行定时任务的线程池

//创建单个线程池
   ScheduledExecutorService scheduledExecutorService= Executors.newSingleThreadScheduledExecutor();
    scheduledExecutorService.scheduleWithFixedDelay(new Runnable() {
          @Override
          public void run() {
             System.out.println("执行任务"+new Date());
           }
       },1,3, TimeUnit.SECONDS);

⑤创建单个线程池

//创建单个线程的线程池
   ExecutorService service= Executors.newSingleThreadScheduledExecutor(); //同步执行
     for (int i = 0; i < 10; i++) {
        service.execute(new Runnable() {
           @Override
           public void run() {
              System.out.println("线程名:"+Thread.currentThread().getName());
                }
            });
        }

经典面试题:创建单个线程池有什么用?

1.可以避免频繁的创建和销毁线程带来的性能开销。
2.有任务队列可以存储多余的任务。
3.当大量的任务不能处理的时候,可以友好的执行拒绝策略。
4.线程池可以很好的管理任务。

⑥根据当前的硬件CPU生成对应个数的线程池,并且是异步处理。(JDK8+)

最大生成的线程数=当前电脑的CPU核数
在这里插入图片描述

//创建一个异步根据当前CPU产生的线程池
      ExecutorService service= Executors.newWorkStealingPool(); //异步处理
        for (int i = 0; i < 10; i++) {
            service.execute(new Runnable() {
                @Override
                public void run() {
                    System.out.println("线程名:"+Thread.currentThread().getName());
                }
            });
        }
        //等待线程池执行完成
        while (!service.isTerminated()){
        }

如果使用前6种创建线程池的方式会导致的问题:

1.线程数量不可控(比如创建带缓存的线程池)
2.工作任务量不可控(Integer.MAX_VALUE),可能会导致内存溢出(OOM)。

2.原始的线程池创建方式ThreadPoolExecutor()

(1)代码实现

public class ThreadDemo {
    private static SimpleDateFormat simpleDateFormat=new SimpleDateFormat("mm:ss");
    public static void main(String[] args) {
        //定义线程池
        ThreadPoolExecutor threadPoolExecutor=new ThreadPoolExecutor(10,10,60,
                TimeUnit.SECONDS,new LinkedBlockingDeque<>(1000));
        for (int i = 0; i < 1001; i++) {
            final int finalI=i;
            threadPoolExecutor.submit(new Runnable() {
                @Override
                public void run() {
                    Date date=new Date(finalI*1000);
                    myFormatTime(date);
                }
            });
        }
    }
    private synchronized static void myFormatTime(Date date){
        String result=simpleDateFormat.format(date);
        System.out.println("时间:"+result);
    }
}

在这里插入图片描述注意事项: 最大的线程数必须大于等于核心线程数

(2)线程池的拒绝策略

在这里插入图片描述

(3)ThreadPoolExecutor执行流程:核心线程数、最大线程数、任务队列

在这里插入图片描述在这里插入图片描述

(4)线程池执行的2种方式

  • 1.执行任务无返回值:excute(new Runnable(){…});
  • 2.执行任务有返回值:submit(Runnable 无返回值/Callable 有返回值)

(5)线程池终止的2种方式

//结束线程池
threadPoolExecutor.shutdown();
//立即终止线程池(线程池的任务不会执行完)
threadPoolExecutor.shutdownNow();

(6)线程池的状态(线程的状态不等于线程池的状态)

在这里插入图片描述

  • 0
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
当 http-nio-7001-exec- 线程池量突然增高时,可能是由以下原因导致的: 1. 突发流量:如果有大量用户同时访问网站或请求需要处理大量据,可能会导致线程池量突然增加。 2. 长时间运行的请求:如果请求需要很长时间才能处理完毕,线程池可能会一直保持高水平。这可能是由于处理请求的代码有性能问题。 3. 竞态条件:当不同的线程需要共享据时,可能会出现竞争条件。这可能会导致某些线程等待其他线程完成工作,从而导致线程池量增加。 4. 内存泄漏:如果应用程序中存在内存泄漏,可能会导致线程池量增加。这是因为线程需要占用内存,而内存泄漏则会导致内存无法释放,从而导致线程不断增加。 要解决此问题,可以通过以下方式: 1. 检查代码逻辑,查找潜在的性能问题,例如避免使用锁、减少据访问等。 2. 应该检查应用程序中的垃圾回收机制和内存分配,以确保内存使用状况正常。 3. 可以增加线程池的大小,以便处理更多的请求。虽然这并不能解决问题的根源,但如果您确定出现了时间敏感的性能问题,则可以使用此方法暂时缓解问题。 4. 内存泄漏是一个严重的问题,应该彻底检查应用程序和底层框架,以确保它们正常运行。 总之,应该确保线程池不会过度使用资源,且需要密切监视线程池并进行必要的调整以支持流量变化。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

秃头小宝儿

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值