【多线程】线程池总结带你详细了解线程池

简介: 【多线程】线程池总结带你详细了解线程池

线程池

线程池是一种线程使用模式。线程过多会带来调度开销,进而影响缓存局部性和整体性能。而线程池维护着多个线程,等待着监督管理者分配可并发执行的任务。避免了在处理短时间任务时创建与销毁线程的代价。线程池不仅能够保证内核的充分利用,还能防止过分调度。线程池最大的好处是减少了每次创建销毁线程的开销。

标准库中的线程池

Executors 创建线程池的几种方式

① 使用 Executors.newFixedThreadPool(5) 能创建出固定包含 5个线程的线程池

返回值类型为 ExecutorService

通过 ExecutorService.submit 可以注册一个任务到线程池中

import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
public class ThreadPool01 {
    public static void main(String[] args) {
        ExecutorService pool = Executors.newFixedThreadPool(5);
        for (int i = 0; i < 20; i++) {
            pool.submit(new Runnable() {
                @Override
                public void run() {
                    System.out.println(Thread.currentThread().getName());
                }
            });
        }
    }
}

② 使用 Executors.newCachedThreadPool()创建出的线程池对象的特点是线程池数目能够动态适应,随着往线程池里添加任务,这个线程池里的线程数量会根据需要自动被创建,创建之后也不会着急销毁,会在线程池里存在一段时间以备随时使用。

import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
public class ThreadPool2 {
    public static void main(String[] args) {
        ExecutorService pool = Executors.newCachedThreadPool();
        for (int i = 0; i < 20; i++) {
            pool.submit(new Runnable() {
                @Override
                public void run() {
                    System.out.println(Thread.currentThread().getName());
                }
            });
        }
    }
}

③使用 Executors.newSingleThreadExecutor(): 创建只包含单个线程的线程池(一般用的不多)

import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
public class ThreadPool3 {
    public static void main(String[] args) {
        ExecutorService pool = Executors.newSingleThreadExecutor();
        for (int i = 0; i < 20; i++) {
            pool.submit(new Runnable() {
                @Override
                public void run() {
                    System.out.println(Thread.currentThread().getName());
                }
            });
        }
    }
}

④使用 Executors.newScheduledThreadPool():相当于是定时器 设定 延迟时间后执行命令,或者定期执行命令

import java.util.concurrent.Executors;
import java.util.concurrent.ExecutorService;
public class TheadPool4 {
    public static void main(String[] args) {
        ExecutorService pool = Executors.newScheduledThreadPool(5);
        for (int i = 0; i < 20; i++) {
            pool.submit(new Runnable() {
                @Override
                public void run() {
                    System.out.println(Thread.currentThread().getName());
                }
            });
        }
    }
}

Executors 本质上是 ThreadPoolExecutor 类的封装,ThreadPoolExecutor 提供了更多的可选参数, 可以进一步细化线程池行为的设定。

ThreadPoolExecutor创建线程池

最后一个构造方法可以涵盖上面其他三个,所以我们重点来看最后一个构造方法。

参数详解:

corePoolSize:核心线程数------>我们如果把线程池理解成一个公司,核心线程数就相当于正式员工数量

maximumPoolSize:最大线程数------>正式员工 + 实习生

keepAliveTime:空余线程等待新任务的最大时间------>允许实习生摸鱼的最大时间(如果说正式员工可以摸鱼并且不会被开除,实习生不允许摸鱼)

unit:keepAliveTime的时间单位

workQueue:阻塞队列,用来存放线程池中的任务,可以根据需要来设置这里的队列是啥比如:如果需要优先级则可以设置为PriorityBlockingQueue 如果不需要优先级线程数目是恒定的则可以使用ArrayBlockingQueue 如果任务变动较大可以使用LinkedBlockingQueue

threadFactory:工厂模式的体现,这里使用threadFactory作为工厂类,由这个类负责创建线程,使用工厂类创建线程主要是为了在创建过程中可以很方便对线程的属性进行设置

handler:线程池的拒绝策略,一个线程池能够容纳的任务数量有上限,当持续向线程池中添加任务,一旦达到上限,不同的拒绝策略有不同的效果

拒绝策略:↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓

问题讨论:使用线程池,设置多少线程数目合适呢???

解释:一个线程执行的代码主要有两类:①CPU密集型:代码主要是进行算数运算/逻辑判断 ②IO密集型:代码主要进行的是IO操作

假设一个线程的所有代码都是CPU密集型代码,这时线程池数量不应该超过N(CPU逻辑核心数),即使超过N也无法提高效率,CPU占满了增加线程反而会增加调度的开销。

假设一个线程的所有代码都是IO密集型,这时不占CPU,此时设置线程数就可以超过N,一个核心可以通过调度来并发执行。

但是纯CPU密集型和纯IO密集型代码一般工作中不会出现,所以线程池的线程数量应该对程序进行测试。

模拟实现线程池

模拟实现线程池我们需要知道线程池里有什么:

1、线程池管理器(ThreadPoolManager):用于创建并管理线程池

2、工作线程(WorkThread): 线程池中线程

3、任务接口(Task):每个任务必须实现的接口,以供工作线程调度任务的执行

4、任务队列:用于存放没有处理的任务。提供一种缓冲机制

核心操作为 submit, 将任务加入线程池中,使用一个 BlockingQueue 组织所有的任务 。

模拟实现一个简单版的线程池👇👇👇

import java.util.concurrent.ArrayBlockingQueue;
import java.util.concurrent.BlockingQueue;
//模拟实现线程池
public class MyThreadPool {
    //定义一个任务队列
    private BlockingQueue<Runnable> deque = new ArrayBlockingQueue<>(1000);
    //把任务添加到队列中
    public void submit(Runnable runnable) throws InterruptedException {
        deque.put(runnable);
    }
    public MyThreadPool(int n) {
        //创建出n个线程负责上述队列中的任务
        for (int i = 0; i < n; i++) {
            Thread t = new Thread(() -> {
                //让线程消费任务并执行
                try {
                    Runnable runnable = deque.take();
                    runnable.run();
                } catch (InterruptedException e) {
                    throw new RuntimeException(e);
                }
            });
            t.start();
        }
    }
}
相关文章
|
18天前
|
监控 安全 Java
在 Java 中使用线程池监控以及动态调整线程池时需要注意什么?
【10月更文挑战第22天】在进行线程池的监控和动态调整时,要综合考虑多方面的因素,谨慎操作,以确保线程池能够高效、稳定地运行,满足业务的需求。
96 38
|
16天前
|
Java
线程池内部机制:线程的保活与回收策略
【10月更文挑战第24天】 线程池是现代并发编程中管理线程资源的一种高效机制。它不仅能够复用线程,减少创建和销毁线程的开销,还能有效控制并发线程的数量,提高系统资源的利用率。本文将深入探讨线程池中线程的保活和回收机制,帮助你更好地理解和使用线程池。
42 2
|
18天前
|
Prometheus 监控 Cloud Native
JAVA线程池监控以及动态调整线程池
【10月更文挑战第22天】在 Java 中,线程池的监控和动态调整是非常重要的,它可以帮助我们更好地管理系统资源,提高应用的性能和稳定性。
54 4
|
18天前
|
Prometheus 监控 Cloud Native
在 Java 中,如何使用线程池监控以及动态调整线程池?
【10月更文挑战第22天】线程池的监控和动态调整是一项重要的任务,需要我们结合具体的应用场景和需求,选择合适的方法和策略,以确保线程池始终处于最优状态,提高系统的性能和稳定性。
80 2
|
21天前
|
Java 开发者
在Java多线程编程中,创建线程的方法有两种:继承Thread类和实现Runnable接口
【10月更文挑战第20天】在Java多线程编程中,创建线程的方法有两种:继承Thread类和实现Runnable接口。本文揭示了这两种方式的微妙差异和潜在陷阱,帮助你更好地理解和选择适合项目需求的线程创建方式。
15 3
|
21天前
|
Java 开发者
在Java多线程编程中,选择合适的线程创建方法至关重要
【10月更文挑战第20天】在Java多线程编程中,选择合适的线程创建方法至关重要。本文通过案例分析,探讨了继承Thread类和实现Runnable接口两种方法的优缺点及适用场景,帮助开发者做出明智的选择。
14 2
|
21天前
|
Java
Java中多线程编程的基本概念和创建线程的两种主要方式:继承Thread类和实现Runnable接口
【10月更文挑战第20天】《JAVA多线程深度解析:线程的创建之路》介绍了Java中多线程编程的基本概念和创建线程的两种主要方式:继承Thread类和实现Runnable接口。文章详细讲解了每种方式的实现方法、优缺点及适用场景,帮助读者更好地理解和掌握多线程编程技术,为复杂任务的高效处理奠定基础。
27 2
|
21天前
|
Java 开发者
Java多线程初学者指南:介绍通过继承Thread类与实现Runnable接口两种方式创建线程的方法及其优缺点
【10月更文挑战第20天】Java多线程初学者指南:介绍通过继承Thread类与实现Runnable接口两种方式创建线程的方法及其优缺点,重点解析为何实现Runnable接口更具灵活性、资源共享及易于管理的优势。
27 1
|
1月前
|
存储 消息中间件 资源调度
C++ 多线程之初识多线程
这篇文章介绍了C++多线程的基本概念,包括进程和线程的定义、并发的实现方式,以及如何在C++中创建和管理线程,包括使用`std::thread`库、线程的join和detach方法,并通过示例代码展示了如何创建和使用多线程。
40 1
C++ 多线程之初识多线程
|
21天前
|
安全 Java 开发者
Java多线程中的`wait()`、`notify()`和`notifyAll()`方法,探讨了它们在实现线程间通信和同步中的关键作用
本文深入解析了Java多线程中的`wait()`、`notify()`和`notifyAll()`方法,探讨了它们在实现线程间通信和同步中的关键作用。通过示例代码展示了如何正确使用这些方法,并分享了最佳实践,帮助开发者避免常见陷阱,提高多线程程序的稳定性和效率。
31 1