java线程池超详细解析

本文涉及的产品
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
全局流量管理 GTM,标准版 1个月
云解析 DNS,旗舰版 1个月
简介: java线程池超详细解析

一、线程池主要核心原理

  • 创建一个池子,池子中是空的
  • 提交任务时,池子会创建新的线程对象,任务执行完毕,线程归还给池子下回再次提交任务时,不需要创建新的线程,直接复用已有的线程即可
  • 但是如果提交任务时,池子中没有空闲线程,也无法创建新的线程,任务就会排队等待

二、线程池代码实现

Executors:线程池的工具类,通过调用方法返回不同类型的线程池对象

  • public static ExecutorService newCachedThreadPool(); 创建一个没有上限的线程池
  • public static ExecutorService newFixedThreadPool(int nThreads); 创建有上限的线程池

具体实现步骤

1、创建线程池

2、提交任务

3、所有的任务全部执行完毕,关闭线程池

执行任务MyRunnable类

package com.hidata.devops.paas.demo;
/**
 * @Description :
 * @Date: 2023-10-11 09:41
 */
public class MyRunnable implements Runnable {
    @Override
    public void run() {
        System.out.println(Thread.currentThread().getName()  );
    }
}

测试类

import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
/**
 * @Description :
 * @Date: 2023-10-11 09:42
 */
public class TestDemo {
    public static void main(String[] args) {
      //创建线程池对象
        ExecutorService executorService = Executors.newFixedThreadPool(3);
    //提交任务
        executorService.submit(new MyRunnable());
        executorService.submit(new MyRunnable());
        executorService.submit(new MyRunnable());
        executorService.submit(new MyRunnable());
        executorService.submit(new MyRunnable());
        executorService.submit(new MyRunnable());
        executorService.submit(new MyRunnable());
        executorService.submit(new MyRunnable());
    //关闭线程池(一般不会关闭)
        executorService.shutdown();
    }
}

运行结果

pool-1-thread-1
pool-1-thread-3
pool-1-thread-2
pool-1-thread-1
pool-1-thread-2
pool-1-thread-3
pool-1-thread-2
pool-1-thread-1
Process finished with exit code 0

总结

  • 上述代码中,我们创建的线程池,设置的最大线程数为3,当连续提交了8个任务,可以看到打印的结果,只有三个线程在轮换着执行任务,所以当任务数超过最大线程数时,线程池不会创建新的线程,而是等线程池中有空闲的线程时,会复用该空闲线程,继续执行任务,再次强调一遍:不会创建新线程,因为最大线程数为3
  • 实际开发中,线程池一般不会关闭哦

三、自定义线程池

通过Executors工具类调用方法,创建线程池,不够灵活,因为很多的线程池参数都没有办法自定义。所以我们需要自定义线程池。

1、自定义线程池原理

1、创建一个空的池子

2、有任务提交时,线程池会创建线程去执行任务,执行完毕归还线程

不断地提交任务,会有以下三个临界点:

1、当核心线程满时,再提交任务就会排队

2、当核心线程满,队伍满时,会创建临时线程

3、当核心线程满,队伍满,临时线程满时,会触发任务拒绝策略

2、自定义线程池参数

3、自定义线程池任务拒绝策略

3、代码实现

package com.hidata.devops.paas.demo;
/**
 * @Description :
 * @Date: 2023-10-11 09:41
 */
public class MyRunnable implements Runnable {
    @Override
    public void run() {
        System.out.println(Thread.currentThread().getName()  );
    }
}
package com.hidata.devops.paas.demo;
import java.util.concurrent.*;
/**
 * @Description :
 * @Date: 2023-10-11 09:42
 */
public class TestDemo {
    public static void main(String[] args) {
        ThreadPoolExecutor executor = new ThreadPoolExecutor(
                3,//核心线程数量,不能小于0
                6,//最大线程数,不能小于0,最大数量>=核心线程数量
                60,//空闲线程最大存活时间
                TimeUnit.SECONDS,//时间单位
                new ArrayBlockingQueue<>(3),//任务队列
                Executors.defaultThreadFactory(),//创建线程工厂
                new ThreadPoolExecutor.AbortPolicy()//任务拒绝策略
        );
        executor.submit(new MyRunnable());
        executor.submit(new MyRunnable());
        executor.submit(new MyRunnable());
        executor.submit(new MyRunnable());
        executor.submit(new MyRunnable());
        executor.submit(new MyRunnable());
        executor.submit(new MyRunnable());
        executor.submit(new MyRunnable());
        executor.shutdown();
    }
}

运行结果

pool-1-thread-2
pool-1-thread-4
pool-1-thread-3
pool-1-thread-3
pool-1-thread-1
pool-1-thread-4
pool-1-thread-2
pool-1-thread-5
Process finished with exit code 0

四、线程池多大合适呢?

线程池大小不能随便写的,要根据你项目的类型,一般分为两种类型的项目:

  • CPU密集型项目:一般是你项目中的计算比较多,但是读取数据库或者读取本地文件操作比较少
  • I/O密集型项目:读取数据库操作或者读取文件操作比较多

针对这两种类型的项目,是有一个公式的,先看一下这个复杂的公式:

我们发现公式里面出现了多个关键字:最大并行数,那怎么获取操作系统的最大并行数呢?

java底层虚拟机已经提供了获取操作系统最大并行数的方法了:

int count = Runtime.getRuntime().availableProcessors();
System.out.println(count);

也可以打开你电脑的任务管理器,其中逻辑处理器就是最大并行数


相关文章
|
1天前
|
监控 Java API
Java并发编程之线程池深度解析
【7月更文挑战第14天】在Java并发编程领域,线程池是提升性能、管理资源的关键工具。本文将深入探讨线程池的核心概念、内部工作原理以及如何有效使用线程池来处理并发任务,旨在为读者提供一套完整的线程池使用和优化策略。
|
4天前
|
缓存 安全 Java
Java中线程池如何管理?
【7月更文挑战第11天】Java中线程池如何管理?
9 2
|
4天前
|
监控 Java 调度
Java面试题:描述Java线程池的概念、用途及常见的线程池类型。介绍一下Java中的线程池有哪些优缺点
Java面试题:描述Java线程池的概念、用途及常见的线程池类型。介绍一下Java中的线程池有哪些优缺点
18 1
|
4天前
|
设计模式 存储 安全
Java面试题:设计一个线程安全的单例类并解释其内存占用情况?使用Java多线程工具类实现一个高效的线程池,并解释其背后的原理。结合观察者模式与Java并发框架,设计一个可扩展的事件处理系统
Java面试题:设计一个线程安全的单例类并解释其内存占用情况?使用Java多线程工具类实现一个高效的线程池,并解释其背后的原理。结合观察者模式与Java并发框架,设计一个可扩展的事件处理系统
13 1
|
4天前
|
设计模式 安全 Java
Java面试题:请解释Java中的线程池以及为什么要使用线程池?请解释Java中的内存模型以及如何避免内存泄漏?请解释Java中的并发工具包以及如何实现一个简单的线程安全队列?
Java面试题:请解释Java中的线程池以及为什么要使用线程池?请解释Java中的内存模型以及如何避免内存泄漏?请解释Java中的并发工具包以及如何实现一个简单的线程安全队列?
10 1
|
4天前
|
设计模式 缓存 安全
Java面试题:工厂模式与内存泄漏防范?线程安全与volatile关键字的适用性?并发集合与线程池管理问题
Java面试题:工厂模式与内存泄漏防范?线程安全与volatile关键字的适用性?并发集合与线程池管理问题
10 1
|
4天前
|
缓存 Java
Java面试题:描述Java中的线程池及其实现方式,详细说明其原理
Java面试题:描述Java中的线程池及其实现方式,详细说明其原理
6 0
|
4天前
|
设计模式 并行计算 安全
Java面试题:如何使用设计模式优化多线程环境下的资源管理?Java内存模型与并发工具类的协同工作,描述ForkJoinPool的工作机制,并解释其在并行计算中的优势。如何根据任务特性调整线程池参数
Java面试题:如何使用设计模式优化多线程环境下的资源管理?Java内存模型与并发工具类的协同工作,描述ForkJoinPool的工作机制,并解释其在并行计算中的优势。如何根据任务特性调整线程池参数
10 0
|
5天前
|
存储 安全 Java
Java面试题:Java内存管理、多线程与并发框架:一道综合性面试题的深度解析,描述Java内存模型,并解释如何在应用中优化内存使用,阐述Java多线程的创建和管理方式,并讨论线程安全问题
Java面试题:Java内存管理、多线程与并发框架:一道综合性面试题的深度解析,描述Java内存模型,并解释如何在应用中优化内存使用,阐述Java多线程的创建和管理方式,并讨论线程安全问题
8 0
|
26天前
|
XML Java 数据格式
深度解析 Spring 源码:从 BeanDefinition 源码探索 Bean 的本质
深度解析 Spring 源码:从 BeanDefinition 源码探索 Bean 的本质
28 3

推荐镜像

更多