java基础-线程池讲解

简介: 你好看官,里面请!今天笔者讲的是线程池。不懂或者觉得我写的有问题可以在评论区留言,我看到会及时回复。 注意:本文仅用于学习参考,不可用于商业用途,如需转载请跟我联系

线程池

使用线程池有哪些优势

1:线程和任务分离,提升线程重用性;

2:控制线程并发数量,降低服务器压力,统一管理所有线程;

3:提升系统响应速度,假如创建线程用的时间为T1,执行任务用的时间为T2,销毁线程用的时间为T3,那么使用线程池就免去了T1和T3的时间;(复用线程)

Runnable接口的弊端

Runnable新建线程,性能差

线程缺乏统一管理,可能无限制的新建线程,相互竞争,严重时会占用过多系统资源导致死机或内存溢出

ThreadPool线程池

重用存在的线程,减少线程对象创建、消亡的开销

线程总数可控,提高资源的利用率

提供额外功能,定时执行、定期执行、监控等

四种线程池

JUC支持的线程池种类

在java.util.concurrent中,提供了工具类Executors(调度 器)对象来创建线程池,可创建的线程池有四种:

1. FixedThreadPool-定长线程池

2. CachedThreadPool-可缓存线程池

3. SingleThreadExecutor-单线程池

4. ScheduledThreadPool-调度线程池

定长线程池( FixedThreadPool)

创建一个指定大小的线程池,可控制线程的最大并发数,超出的线程会在LinkedBlockingQueue阻塞队列中等待

importjava.util.concurrent.ExecutorService;
importjava.util.concurrent.Executors;
publicclassThreadPoolSample1 {
publicstaticvoidmain(String[] args) {
//创建一个可创建一个定长线程池//定长线程池的特点是固定线程总数,空闲线程用于执行任务,如果线程都在使用,后续任务则处于等待状态ExecutorServicees=Executors.newFixedThreadPool(10);
for(inti=1; i<=50 ; i++){
finalintindex=i;
//不需要返回值,使用execute方法执行Runnable对象//可以使用匿名类简化代码es.execute(newRunnable() {
@Overridepublicvoidrun() {
System.out.println(Thread.currentThread().getName()+":"+index );
                }
            });
/*需要返回值,使用submit方法执行Callable对象,利用Future对象接收返回值Future<Object> ret = es.submit(new Callable<Object>() {@Overridepublic Object call() throws Exception {return null;}});*/        }
//处理完毕关闭线程池es.shutdown();
    }
}

可缓存线程池(CachedThreadPool)

创建一个可缓存的无界线程池,如果线程池长度超过处理需要,可灵活回收空线程,若无可回收,则新建线程。当线程池中的线程空闲时间超过60s,则会自动回收该线程,当任务超过线程池的线程数则创建新的线程,线程池的大小上限为Integer.MAX_VALUE,可看作无限大。

importjava.util.concurrent.ExecutorService;
importjava.util.concurrent.Executors;
/*演示*/publicclassThreadPoolSample2 {
publicstaticvoidmain(String[] args) {
//调度器对象//ExecutorService用于管理线程池ExecutorServicethreadPool=Executors.newCachedThreadPool();//创建一个可缓存线程池//可缓存线程池的特点是,无限大,如果线程池中没有可用的线程则创建,有空闲线程则利用起来for(inti=1 ; i<=1000 ; i++) {
finalintindex=i;
threadPool.execute(newRunnable() {
@Overridepublicvoidrun() {
System.out.println(Thread.currentThread().getName() +":"+index);
                }
            });
        }
threadPool.shutdown();
    }
}

可缓存线程池(CachedThreadPool)

创建一个可缓存的无界线程池,如果线程池长度超过处理需要,可灵活回收空线程,若无可回收,则新建线程。当线程池中的线程空闲时间超过60s,则会自动回收该线程,当任务超过线程池的线程数则创建新的线程,线程池的大小上限为Integer.MAX_VALUE,可看作无限大。

importjava.util.concurrent.ExecutorService;
importjava.util.concurrent.Executors;
/*演示*/publicclassThreadPoolSample2 {
publicstaticvoidmain(String[] args) {
//调度器对象//ExecutorService用于管理线程池ExecutorServicethreadPool=Executors.newCachedThreadPool();//创建一个可缓存线程池//可缓存线程池的特点是,无限大,如果线程池中没有可用的线程则创建,有空闲线程则利用起来for(inti=1 ; i<=1000 ; i++) {
finalintindex=i;
threadPool.execute(newRunnable() {
@Overridepublicvoidrun() {
System.out.println(Thread.currentThread().getName() +":"+index);
                }
            });
        }
threadPool.shutdown();
    }
}

单线程池(SingleThreadExecutor)

创建一个单线程化的线程池,它只有一个线程,用仅有的一个线程来执行任务,保证所有的任务按照指定顺序(FIFO,LIFO,优先级)执行,所有的任务都保存在队列LinkedBlockingQueue中,等待唯一的单线程来执行任务。

importjava.util.concurrent.ExecutorService;
importjava.util.concurrent.Executors;
publicclassThreadPoolSample3 {
publicstaticvoidmain(String[] args) {
//调度器对象//ExecutorService用于管理线程池ExecutorServicethreadPool=Executors.newSingleThreadExecutor();//单线程线程池for(inti=1 ; i<=1000 ; i++) {
finalintindex=i;
threadPool.execute(newRunnable() {
@Overridepublicvoidrun() {
System.out.println(Thread.currentThread().getName() +":"+index);
                }
            });
        }
threadPool.shutdown();
    }
}

调度线程池(ScheduledThreadPool)

importjava.util.Date;
importjava.util.concurrent.Executors;
importjava.util.concurrent.ScheduledExecutorService;
importjava.util.concurrent.TimeUnit;
publicclassThreadPoolSample4 {
publicstaticvoidmain(String[] args) {
//调度线程池ScheduledExecutorServicescheduledThreadPool=Executors.newScheduledThreadPool(5);//可调度线程池//初始延迟1秒执行,每三秒执行一次scheduledThreadPool.scheduleAtFixedRate(newRunnable() {
@Overridepublicvoidrun() {
System.out.println(newDate() +"延迟1秒执行,每三秒执行一次");
            }
        }, 1, 3, TimeUnit.SECONDS);
    }
}
相关文章
|
11天前
|
监控 安全 Java
在 Java 中使用线程池监控以及动态调整线程池时需要注意什么?
【10月更文挑战第22天】在进行线程池的监控和动态调整时,要综合考虑多方面的因素,谨慎操作,以确保线程池能够高效、稳定地运行,满足业务的需求。
88 38
|
8天前
|
安全 Java
java 中 i++ 到底是否线程安全?
本文通过实例探讨了 `i++` 在多线程环境下的线程安全性问题。首先,使用 100 个线程分别执行 10000 次 `i++` 操作,发现最终结果小于预期的 1000000,证明 `i++` 是线程不安全的。接着,介绍了两种解决方法:使用 `synchronized` 关键字加锁和使用 `AtomicInteger` 类。其中,`AtomicInteger` 通过 `CAS` 操作实现了高效的线程安全。最后,通过分析字节码和源码,解释了 `i++` 为何线程不安全以及 `AtomicInteger` 如何保证线程安全。
java 中 i++ 到底是否线程安全?
|
13天前
|
Java 调度
[Java]线程生命周期与线程通信
本文详细探讨了线程生命周期与线程通信。文章首先分析了线程的五个基本状态及其转换过程,结合JDK1.8版本的特点进行了深入讲解。接着,通过多个实例介绍了线程通信的几种实现方式,包括使用`volatile`关键字、`Object`类的`wait()`和`notify()`方法、`CountDownLatch`、`ReentrantLock`结合`Condition`以及`LockSupport`等工具。全文旨在帮助读者理解线程管理的核心概念和技术细节。
30 1
[Java]线程生命周期与线程通信
|
3天前
|
存储 设计模式 分布式计算
Java中的多线程编程:并发与并行的深度解析####
在当今软件开发领域,多线程编程已成为提升应用性能、响应速度及资源利用率的关键手段之一。本文将深入探讨Java平台上的多线程机制,从基础概念到高级应用,全面解析并发与并行编程的核心理念、实现方式及其在实际项目中的应用策略。不同于常规摘要的简洁概述,本文旨在通过详尽的技术剖析,为读者构建一个系统化的多线程知识框架,辅以生动实例,让抽象概念具体化,复杂问题简单化。 ####
|
4天前
|
Java 开发者
在Java多线程编程的世界里,Lock接口正逐渐成为高手们的首选,取代了传统的synchronized关键字
在Java多线程编程的世界里,Lock接口正逐渐成为高手们的首选,取代了传统的synchronized关键字
19 4
|
4天前
|
消息中间件 供应链 Java
掌握Java多线程编程的艺术
【10月更文挑战第29天】 在当今软件开发领域,多线程编程已成为提升应用性能和响应速度的关键手段之一。本文旨在深入探讨Java多线程编程的核心技术、常见问题以及最佳实践,通过实际案例分析,帮助读者理解并掌握如何在Java应用中高效地使用多线程。不同于常规的技术总结,本文将结合作者多年的实践经验,以故事化的方式讲述多线程编程的魅力与挑战,旨在为读者提供一种全新的学习视角。
24 3
|
10天前
|
安全 Java
在 Java 中使用实现 Runnable 接口的方式创建线程
【10月更文挑战第22天】通过以上内容的介绍,相信你已经对在 Java 中如何使用实现 Runnable 接口的方式创建线程有了更深入的了解。在实际应用中,需要根据具体的需求和场景,合理选择线程创建方式,并注意线程安全、同步、通信等相关问题,以确保程序的正确性和稳定性。
|
5天前
|
安全 Java 调度
Java中的多线程编程入门
【10月更文挑战第29天】在Java的世界中,多线程就像是一场精心编排的交响乐。每个线程都是乐团中的一个乐手,他们各自演奏着自己的部分,却又和谐地共同完成整场演出。本文将带你走进Java多线程的世界,让你从零基础到能够编写基本的多线程程序。
17 1
|
9天前
|
缓存 Java 调度
Java中的多线程编程:从基础到实践
【10月更文挑战第24天】 本文旨在为读者提供一个关于Java多线程编程的全面指南。我们将从多线程的基本概念开始,逐步深入到Java中实现多线程的方法,包括继承Thread类、实现Runnable接口以及使用Executor框架。此外,我们还将探讨多线程编程中的常见问题和最佳实践,帮助读者在实际项目中更好地应用多线程技术。
17 3
|
11天前
|
监控 安全 Java
Java多线程编程的艺术与实践
【10月更文挑战第22天】 在现代软件开发中,多线程编程是一项不可或缺的技能。本文将深入探讨Java多线程编程的核心概念、常见问题以及最佳实践,帮助开发者掌握这一强大的工具。我们将从基础概念入手,逐步深入到高级主题,包括线程的创建与管理、同步机制、线程池的使用等。通过实际案例分析,本文旨在提供一种系统化的学习方法,使读者能够在实际项目中灵活运用多线程技术。