【Java 并发编程】线程池机制 ( 线程池示例 | newCachedThreadPool | newFixedThreadPool | newSingleThreadExecutor )

简介: 【Java 并发编程】线程池机制 ( 线程池示例 | newCachedThreadPool | newFixedThreadPool | newSingleThreadExecutor )

文章目录

前言

一、线程池示例

二、newCachedThreadPool 线程池示例

三、newFixedThreadPool 线程池示例

三、newSingleThreadExecutor 线程池示例

前言

在上一篇博客 【Java 并发编程】线程池机制 ( 测试线程开销 | 启动线程分析 | 用户态 | 内核态 | 用户线程 | 内核线程 | 轻量级进程 ) 中 , 分析了线程的开销 ;


本篇博客中使用线程池改造该示例并进行分析 ;






一、线程池示例


创建 10 1010 万线程 , 需要 10992 1099210992 ms ; 使用线程池启动 10 1010 万线程 , 仅需要 26 2626 ms ;


线程池的效率比线程高几个数量级 ;



线程池示例 :


import java.util.concurrent.CountDownLatch;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
public class Main {
    /**
     * 线程中对该值进行累加操作
     */
    private static int count = 0;
    public static void main(String[] args) throws InterruptedException {
        // 记录程序开始执行时间
        long startTime = System.currentTimeMillis();
        // 线程池
        ExecutorService executorService = Executors.newSingleThreadExecutor();
        // 每次线程执行完毕, 计数 -1 , 当计数减到 0 之后, 才能解除阻塞
        CountDownLatch countDownLatch = new CountDownLatch(100000);
        // 创建 10 万个线程, 开启线程后, 向集合中添加一个元素
        for (int i = 0; i < 100000; i ++) {
            executorService.execute(new Runnable() {
                @Override
                public void run() {
                    count ++;
                    // 计数 -1
                    countDownLatch.countDown();
                }
            });
        }
        // 阻塞等待计数 -1, 直到 10000 个线程执行完毕
        //  使用这种方式确定所有线程执行完毕
        countDownLatch.await();
        // 执行完毕后, 停掉线程池, 否则程序不会退出
        executorService.shutdown();
        // 记录程序执行结束时间
        long endTime = System.currentTimeMillis();
        // 打印消耗的时间
        System.out.println("耗时 : " + ( endTime - startTime ) + " ms , 最终 count = " + count);
    }
}


执行结果 : 原来使用线程需要 10992 1099210992 ms 时间 , 使用线程池后 , 仅需要 26 2626 ms , 这效率提升了好几个数量级 ;

image.png



等待线程执行结束 , 直接调用 Thread.join() 方法 , 等待线程池结束 , 借助 CountDownLatch 通过线程计数来确定线程是否执行完毕 ;



调用 ExecutorService executorService = Executors.newSingleThreadExecutor() 创建线程池 , 创建的是 FinalizableDelegatedExecutorService 线程池 ;


public static ExecutorService newSingleThreadExecutor() {
        return new FinalizableDelegatedExecutorService
            (new ThreadPoolExecutor(1, 1,
                                    0L, TimeUnit.MILLISECONDS,
                                    new LinkedBlockingQueue<Runnable>()));
    }






二、newCachedThreadPool 线程池示例


import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
public class Main {
    public static void main(String[] args) {
        ExecutorService executorService1 = Executors.newCachedThreadPool();
        for (int i = 0; i < 100; i ++) {
            executorService1.execute(new Task(i));
        }
    }
    static class Task implements Runnable {
        /**
         * 记录线程的索引 0 ~ 99
         */
        private int i = 0;
        public Task(int i) {
            this.i = i;
        }
        @Override
        public void run() {
            System.out.println("线程 ID : " + Thread.currentThread().getName() + " , 线程索引 : " + i);
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
}


执行结果 : 该线程池中创建了 100 100100 个线程 , 执行 100 100100 个任务 ;


image.png






三、newFixedThreadPool 线程池示例


import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
public class Main {
    public static void main(String[] args) {
        ExecutorService executorService2 = Executors.newFixedThreadPool(10);
        for (int i = 0; i < 100; i ++) {
            executorService2.execute(new Task(i));
        }
    }
    static class Task implements Runnable {
        /**
         * 记录线程的索引 0 ~ 99
         */
        private int i = 0;
        public Task(int i) {
            this.i = i;
        }
        @Override
        public void run() {
            System.out.println("线程 ID : " + Thread.currentThread().getName() + " , 线程索引 : " + i);
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
}


执行结果 : 该线程池中创建了 10 1010 个线程 , 执行 100 100100 个任务 ;


image.png






三、newSingleThreadExecutor 线程池示例


import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
public class Main {
    public static void main(String[] args) {
        ExecutorService executorService3 = Executors.newSingleThreadExecutor();
        for (int i = 0; i < 100; i ++) {
            executorService3.execute(new Task(i));
        }
    }
    static class Task implements Runnable {
        /**
         * 记录线程的索引 0 ~ 99
         */
        private int i = 0;
        public Task(int i) {
            this.i = i;
        }
        @Override
        public void run() {
            System.out.println("线程 ID : " + Thread.currentThread().getName() + " , 线程索引 : " + i);
            try {
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
}


执行结果 : 该线程池中创建了 1 11 个线程 , 执行 100 100100 个任务 ;


image.png

目录
相关文章
|
10天前
|
XML 安全 Java
Java反射机制:解锁代码的无限可能
Java 反射(Reflection)是Java 的特征之一,它允许程序在运行时动态地访问和操作类的信息,包括类的属性、方法和构造函数。 反射机制能够使程序具备更大的灵活性和扩展性
19 5
Java反射机制:解锁代码的无限可能
|
4天前
|
存储 缓存 安全
Java内存模型(JMM):深入理解并发编程的基石####
【10月更文挑战第29天】 本文作为一篇技术性文章,旨在深入探讨Java内存模型(JMM)的核心概念、工作原理及其在并发编程中的应用。我们将从JMM的基本定义出发,逐步剖析其如何通过happens-before原则、volatile关键字、synchronized关键字等机制,解决多线程环境下的数据可见性、原子性和有序性问题。不同于常规摘要的简述方式,本摘要将直接概述文章的核心内容,为读者提供一个清晰的学习路径。 ####
16 2
|
4天前
|
安全 IDE Java
Java反射Reflect机制详解
Java反射(Reflection)机制是Java语言的重要特性之一,允许程序在运行时动态地获取类的信息,并对类进行操作,如创建实例、调用方法、访问字段等。反射机制极大地提高了Java程序的灵活性和动态性,但也带来了性能和安全方面的挑战。本文将详细介绍Java反射机制的基本概念、常用操作、应用场景以及其优缺点。 ## 基本概念 ### 什么是反射 反射是一种在程序运行时动态获取类的信息,并对类进行操作的机制。通过反射,程序可以在运行时获得类的字段、方法、构造函数等信息,并可以动态调用方法、创建实例和访问字段。 ### 反射的核心类 Java反射机制主要由以下几个类和接口组成,这些类
13 2
|
9天前
|
存储 缓存 安全
🌟Java零基础:深入解析Java序列化机制
【10月更文挑战第20天】本文收录于「滚雪球学Java」专栏,专业攻坚指数级提升,希望能够助你一臂之力,帮你早日登顶实现财富自由🚀;同时,欢迎大家关注&&收藏&&订阅!持续更新中,up!up!up!!
18 3
|
9天前
|
安全 Java UED
深入理解Java中的异常处理机制
【10月更文挑战第25天】在编程世界中,错误和意外是不可避免的。Java作为一种广泛使用的编程语言,其异常处理机制是确保程序健壮性和可靠性的关键。本文通过浅显易懂的语言和实际示例,引导读者了解Java异常处理的基本概念、分类以及如何有效地使用try-catch-finally语句来处理异常情况。我们将从一个简单的例子开始,逐步深入到异常处理的最佳实践,旨在帮助初学者和有经验的开发者更好地掌握这一重要技能。
16 2
|
9天前
|
Java
线程池内部机制:线程的保活与回收策略
【10月更文挑战第24天】 线程池是现代并发编程中管理线程资源的一种高效机制。它不仅能够复用线程,减少创建和销毁线程的开销,还能有效控制并发线程的数量,提高系统资源的利用率。本文将深入探讨线程池中线程的保活和回收机制,帮助你更好地理解和使用线程池。
34 2
|
11天前
|
存储 运维 Java
💻Java零基础:深入了解Java内存机制
【10月更文挑战第18天】本文收录于「滚雪球学Java」专栏,专业攻坚指数级提升,希望能够助你一臂之力,帮你早日登顶实现财富自由🚀;同时,欢迎大家关注&&收藏&&订阅!持续更新中,up!up!up!!
21 1
|
5天前
|
Java 开发者
深入理解Java异常处理机制
【10月更文挑战第29天】在Java的世界中,异常处理如同生活的调味品,不可或缺。它确保了程序在遇到错误时不会崩溃,而是优雅地继续运行或者给出提示。本文将带你领略异常处理的奥秘,从基础的try-catch语句到高级的自定义异常,让你在面对程序中的各种“意外”时,能够从容应对。
|
7天前
|
SQL Java
探索Java中的异常处理机制
【10月更文挑战第26天】 在本文中,我们将深入探讨Java编程语言的异常处理机制。通过分析不同类型的异常、异常的捕获与抛出方式,以及如何自定义异常类,读者将能够更好地理解并应用Java中的异常处理机制来提高代码的健壮性和可读性。
16 0
|
29天前
|
存储 消息中间件 资源调度
C++ 多线程之初识多线程
这篇文章介绍了C++多线程的基本概念,包括进程和线程的定义、并发的实现方式,以及如何在C++中创建和管理线程,包括使用`std::thread`库、线程的join和detach方法,并通过示例代码展示了如何创建和使用多线程。
38 1
C++ 多线程之初识多线程