【JAVA进阶篇教学】第五篇:Java多线程编程

news2024/11/20 18:42:12

博主打算从0-1讲解下java进阶篇教学,今天教学第五篇:Java多线程编程。   

在Java编程中,使用多线程可以提高程序的并发性能,但是直接创建和管理线程可能会导致资源浪费和性能下降。Java提供了线程池来管理线程的生命周期和执行任务,有效地提高了并发效率。本文将详细介绍如何使用Java线程池以及如何自定义线程池。

目录

一、线程简介

二、FixedThreadPool

三、CachedThreadPool

四、ScheduledThreadPool

五、SingleThreadPool

六、自定义线程池


一、线程简介

Java通过java.util.concurrent包提供了Executor框架来管理线程池。以下是使用线程池的步骤:

  1. 创建线程池:可以通过Executors工厂类来创建不同类型的线程池。常用的线程池类型包括FixedThreadPool、CachedThreadPool、ScheduledThreadPool、SingleThreadPool等。
  2. 提交任务:使用线程池的execute()或submit()方法提交任务给线程池执行。
  3. 关闭线程池:在不需要线程池时,需要及时关闭以释放资源。可以调用线程池的shutdown()或shutdownNow()方法来关闭线程池。

二、FixedThreadPool

FixedThreadPool是一个固定大小的线程池,它会创建指定数量的线程并保持这些线程的数量不变,即使线程处于空闲状态也不会销毁。

优点

  • 线程数量固定,不会随着任务数量的增加而增加,避免了线程数量过多导致的资源消耗问题。
  • 可以控制线程的最大并发数,保证系统资源不被过度占用。

缺点

  • 固定大小的线程池可能会导致任务排队等待执行,如果任务数量过多,可能会导致性能下降。
  • 线程池大小不可动态调整,如果在某些情况下需要更多的线程来处理任务,就无法满足需求。

案例: 

import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

public class FixedThreadPoolExample {
    public static void main(String[] args) {
        // 创建固定大小的线程池,包含5个线程
        ExecutorService executor = Executors.newFixedThreadPool(5);

        // 提交任务给线程池执行
        for (int i = 0; i < 10; i++) {
            executor.execute(new Task(i));
        }

        // 关闭线程池
        executor.shutdown();
    }

    static class Task implements Runnable {
        private int taskId;

        public Task(int taskId) {
            this.taskId = taskId;
        }

        @Override
        public void run() {
            System.out.println("Task " + taskId + " is running on thread " + Thread.currentThread().getName());
        }
    }
}

三、CachedThreadPool

CachedThreadPool是一个可缓存的线程池,它会根据需要创建新线程,并在旧线程可用时重用它们。如果线程在60秒内没有被使用,则会被终止并移除。

优点

  • 线程池的大小可以根据需要自动调整,无需手动设置线程数量,节省了资源并提高了性能。
  • 可以灵活处理大量的短期任务,避免了长时间等待空闲线程的情况。

缺点

  • 可缓存的线程池会创建大量的线程,如果任务数量过多,可能会导致系统资源耗尽。
  • 对于长时间执行的任务,线程池可能会频繁地创建和销毁线程,增加了系统开销。

案例

import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

public class CachedThreadPoolExample {
    public static void main(String[] args) {
        // 创建可缓存的线程池
        ExecutorService executor = Executors.newCachedThreadPool();

        // 提交任务给线程池执行
        for (int i = 0; i < 10; i++) {
            executor.execute(new Task(i));
        }

        // 关闭线程池
        executor.shutdown();
    }

    static class Task implements Runnable {
        private int taskId;

        public Task(int taskId) {
            this.taskId = taskId;
        }

        @Override
        public void run() {
            System.out.println("Task " + taskId + " is running on thread " + Thread.currentThread().getName());
        }
    }
}

四、ScheduledThreadPool

ScheduledThreadPool是一个定时任务线程池,它可以在指定时间或者周期性地执行任务。

优点

  • 可以定时执行任务或周期性执行任务,非常适合需要按照一定频率执行任务的场景。
  • 线程池的大小可以根据任务数量自动调整,灵活性较高。

缺点

  • 定时任务可能会受到系统时间变更的影响,导致任务执行时间不准确。
  • 如果任务执行时间过长,可能会影响后续任务的执行。

案例

import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.TimeUnit;

public class ScheduledThreadPoolExample {
    public static void main(String[] args) {
        // 创建定时任务线程池,包含3个线程
        ScheduledExecutorService executor = Executors.newScheduledThreadPool(3);

        // 延迟1秒后执行任务,并每隔3秒执行一次
        executor.scheduleAtFixedRate(new Task(), 1, 3, TimeUnit.SECONDS);
    }

    static class Task implements Runnable {
        @Override
        public void run() {
            System.out.println("Task is running on thread " + Thread.currentThread().getName());
        }
    }
}

五、SingleThreadPool

SingleThreadPool是一个单线程的线程池,它只会创建一个工作线程来执行任务,保证所有任务按照指定顺序执行。

优点

  • 只有一个工作线程,保证任务按照指定顺序依次执行,避免了多线程情况下的竞争问题。
  • 线程池的大小固定,不存在线程数量动态调整的问题。

缺点

  • 只有一个线程,无法并行处理多个任务,可能会导致任务执行时间过长。
  • 如果任务出现阻塞或异常,可能会影响后续任务的执行。

案例

import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

public class SingleThreadPoolExample {
    public static void main(String[] args) {
        // 创建单线程的线程池
        ExecutorService executor = Executors.newSingleThreadExecutor();

        // 提交任务给线程池执行
        for (int i = 0; i < 10; i++) {
            executor.execute(new Task(i));
        }

        // 关闭线程池
        executor.shutdown();
    }

    static class Task implements Runnable {
        private int taskId;

        public Task(int taskId) {
            this.taskId = taskId;
        }

        @Override
        public void run() {
            System.out.println("Task " + taskId + " is running on thread " + Thread.currentThread().getName());
        }
    }
}

六、自定义线程池

除了使用Executors提供的线程池之外,我们还可以自定义线程池来满足特定的需求。在自定义线程池时,通常需要考虑以下七个参数:

  1. corePoolSize:核心线程数,即线程池中保持活动状态的最小线程数。
  2. maximumPoolSize:最大线程数,即线程池中允许的最大线程数。
  3. keepAliveTime:线程空闲时间,即当线程池中的线程数量超过corePoolSize时,多余的空闲线程等待新任务的最长时间。
  4. unit:keepAliveTime的时间单位。
  5. workQueue:工作队列,用于保存等待执行的任务。
  6. threadFactory:线程工厂,用于创建新线程。
  7. handler:拒绝策略,用于处理当工作队列已满并且无法继续接受新任务时的情况。

关于拒绝策略:

  1. AbortPolicy(默认策略)

    • 在工作队列已满的情况下,直接抛出RejectedExecutionException异常,阻止任务的执行。这是默认的拒绝策略,意味着线程池无法接受新的任务时,会抛出异常来通知调用者。
  2. CallerRunsPolicy

    • 当工作队列已满时,新提交的任务会由提交任务的线程来执行,即调用者所在的线程直接执行该任务。这样做可以减少任务的提交速度,以便控制任务的执行速度。
  3. DiscardPolicy

    • 当工作队列已满时,会丢弃新提交的任务,而不做任何处理。这意味着新提交的任务将被静默地忽略,不会得到执行,也不会抛出异常。
  4. DiscardOldestPolicy

    • 当工作队列已满时,会丢弃队列中最早提交的任务,然后尝试将新提交的任务添加到工作队列中。这样做可以确保工作队列中始终保留着最新的任务,但可能会丢失一些已提交的任务。

下面是一个自定义线程池的示例代码:

import java.util.concurrent.*;

public class CustomThreadPoolExample {
    public static void main(String[] args) {
        ThreadPoolExecutor executor = new ThreadPoolExecutor(
                2, // corePoolSize
                5, // maximumPoolSize
                10, // keepAliveTime
                TimeUnit.SECONDS, // unit
                new ArrayBlockingQueue<>(10), // workQueue
                Executors.defaultThreadFactory(), // threadFactory
                new ThreadPoolExecutor.AbortPolicy() // handler
        );

        // 提交任务给线程池执行
        for (int i = 0; i < 10; i++) {
            executor.execute(new Task(i));
        }

        // 关闭线程池
        executor.shutdown();
    }

    static class Task implements Runnable {
        private int taskId;

        public Task(int taskId) {
            this.taskId = taskId;
        }

        @Override
        public void run() {
            System.out.println("Task " + taskId + " is running on thread " + Thread.currentThread().getName());
        }
    }
}

在自定义线程池中,我们通过构造方法来设置线程池的参数,包括corePoolSize、maximumPoolSize、keepAliveTime、unit、workQueue、threadFactory和handler。通过合理地配置这些参数,我们可以创建出满足不同需求的线程池。

通过本文的介绍,相信你已经对Java中的线程池有了更深入的理解,并能够灵活地使用和自定义线程池来提高程序的并发性能。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1624096.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【蓝桥杯省赛真题40】python摘苹果 中小学青少年组蓝桥杯比赛 算法思维python编程省赛真题解析

目录 python摘苹果 一、题目要求 1、编程实现 2、输入输出 二、算法分析 三、程序编写 四、程序说明 五、运行结果 六、考点分析 七、 推荐资料 1、蓝桥杯比赛 2、考级资料 3、其它资料 python摘苹果 第十三届蓝桥杯青少年组python编程省赛真题 一、题目要求 &…

浅谈数据模型

1&#xff1a;事实表和维表的概述 前言&#xff1a;数据仓库是一种用于存储和管理大量数据的技术。其中&#xff0c;事实表和维表是数据仓库中的两个重要概念&#xff0c;首先了解一下事实表和维度表 1.事实表&#xff1a;是指用于存储测量“事实数据”的表&#xff0c;事实数…

IE浏览器,文件下载失败,onDownloadProgress方法里报错:无法获取未定义或null引用的属性“getResponseheader“

问题背景&#xff1a; 谷歌、火狐、edge都没有问题&#xff0c;ie11浏览器也没有问题&#xff0c;ie10及以下会报错&#xff0c;无法获取未定义或null引用的属性"getResponseheader 查看代码&#xff0c;getResponseHeader这个方法是在获取进度条的时候使用&#xff0c; …

OpenWRT设置自动获取IP,作为二级路由器

前言 上一期咱们讲了在OpenWRT设置PPPoE拨号的教程&#xff0c;在光猫桥接的模式下&#xff0c;OpenWRT如果不设置PPPoE拨号&#xff0c;就无法正常上网。 OpenWRT设置PPPoE拨号教程 但现在很多新装的宽带&#xff0c;宽带师傅为了方便都会把光猫设置为路由模式。如果你再外…

SpringCloud系列(15)--Eureka自我保护

前言&#xff1a;在上一章节中我们说明了一些关于Eureka的服务发现功能&#xff0c;也用这个功能进行接口的实现&#xff0c;在本章节则介绍一些关于Eureka的自我保护 1、Eureka保护模式概述 保护模式主要用于一组客户端和Eureka Server之间存在网络分区场景下的保护。默认情况…

带你走进不一样的策略模式

欢迎来到我的博客&#xff0c;代码的世界里&#xff0c;每一行都是一个故事 带你走进不一样的策略模式 前言策略模式简介概念解释 策略模式的结构策略模式优点项目实践之bean策略构思业务策略实现策略接口实现策略上下文业务实现 前言 在编程的世界里&#xff0c;每一次按键都…

Golang对接Ldap(保姆级教程:概念搭建实战)

Golang对接Ldap&#xff08;保姆级教程&#xff1a;概念&搭建&实战&#xff09; 最近项目需要对接客户的LDAP服务&#xff0c;于是趁机好好了解了一下。LDAP实际是一个协议&#xff0c;对应的实现&#xff0c;大家可以理解为一个轻量级数据库。用户查询。比如&#xff…

高频前端面试题汇总之Vue篇

1. Vue的基本原理 当一个Vue实例创建时&#xff0c;Vue会遍历data中的属性&#xff0c;用 Object.defineProperty&#xff08;vue3.0使用proxy &#xff09;将它们转为 getter/setter&#xff0c;并且在内部追踪相关依赖&#xff0c;在属性被访问和修改时通知变化。 每个组件实…

【ruoyi-vue】登录解析(后端)

调试登录接口 进入实现类可以有 验证码校验 登录前置校验 用户验证 验证码校验 通过uuid获取redis 中存储的验证码信息&#xff0c;获取后对用户填写的验证码数据进行校验比对 用户验证 1.进入控制器的 /login 方法 2.进入security账号鉴权功能&#xff0c;经过jar内的流…

算法-动态规划专题

文章目录 前言 : 动态规划简述1 . 斐波那契模型1.1 泰波那契数列1.2 最小花费爬楼梯1.3 解码方法 前言 : 动态规划简述 动态规划在当前我们的理解下,其实就是一种变相的递归,我们查看一些资料也可以知道,动态规划其实属于递归的一个分支,通过把递归问题开辟的栈帧通过一定的手…

代码随想录第45天|70. 爬楼梯 (进阶)322. 零钱兑换

70. 爬楼梯 &#xff08;进阶&#xff09; 57. 爬楼梯&#xff08;第八期模拟笔试&#xff09; (kamacoder.com) 代码随想录 (programmercarl.com) 假设你正在爬楼梯。需要 n 阶你才能到达楼顶。 每次你可以爬至多m (1 < m < n)个台阶。你有多少种不同的方法可以爬到楼顶…

企业数字化转型,“业务”先行

在当今时代&#xff0c;数字化转型已经成为企业发展的必经之路。数字化转型&#xff0c;简而言之&#xff0c;就是运用数字技术&#xff0c;对企业运营管理的各个环节进行深度改造&#xff0c;以提升企业的运营效率和市场竞争力。据有关机构研究测算&#xff0c;数字化转型可使…

【静态分析】静态分析笔记07 - 指针分析基础

参考&#xff1a; 【课程笔记】南大软件分析课程7——指针分析基础&#xff08;课时9/10&#xff09; - 简书 -------------------------------------------------------------- 1. 指针分析规则 规则&#xff1a;采用推导形式&#xff0c;横线上面是条件&#xff0c;横线下…

Linux开发板配置静态IP

1、查看网口信息&#xff0c;易知eth0无IP地址 ifconfig2、首先分配一个IP地址 sudo ifconfig eth0 192.168.5.8 up3、此时配置的IP地址只是临时的&#xff0c;当你reboot重启板子上电后&#xff0c;ip地址会消失&#xff0c;因此需要为板子配置静态ip&#xff0c;避免每次上…

说方法不如传授经验向媒体投稿你可以这样

在信息爆炸的时代,作为单位的信息宣传员,肩负着将本单位的重要资讯、活动成果、政策解读等内容有效传播至公众视野的重任。其中,向各类媒体投稿无疑是实现这一目标的重要途径。然而,传统的邮件投稿方式常常让我深感力不从心,费时费力不说,成功率低、出稿慢等问题更是让我和领导…

SQL注入简单总结

一、SQL注入是什么 SQL注入即&#xff1a;是指web应用程序对用户输入数据的合法性没有判断或过滤不严&#xff0c;攻击者可以在web应用程序中事先定义好的查询语句的结尾上添加额外的SQL语句&#xff0c;在管理员不知情的情况下实现非法操作&#xff0c;以此来实现欺骗数据库服…

【论文推导】基于有功阻尼的转速环PI参数整定分析

前言 在学习电机控制的路上&#xff0c;PMSM的PI电流控制是不可避免的算法之一&#xff0c;其核心在于内环电流环、外环转速环的设置&#xff0c;来保证转速可调且稳定&#xff0c;并且保证较好的动态性能。整个算法仿真在《现代永磁同步电机控制原理及matlab仿真》中已详细给出…

前端项目中使用插件prettier/jscodeshift/json-stringify-pretty-compact格式化代码或json数据

同学们可以私信我加入学习群&#xff01; 正文开始 前言一、json代码格式化-选型二、json-stringify-pretty-compact简单试用三、prettier在前端使用四、查看prettier支持的语言和插件五、使用prettier格式化vue代码最终效果如图&#xff1a; ![在这里插入图片描述](https://im…

3.8设计模式——State 状态模式(行为型)

意图 允许一个对象在其内部状态改变时改变它的行为。对象看起来似乎修改了它的类。 结构 Context&#xff08;上下文&#xff09;定义客户感兴趣的接口&#xff1b;维护一个ConcreteState子类的实例&#xff0c;这个实例定义当前状态。State&#xff08;状态&#xff09;定义…

[x86] OpenBMC简介

什么是 OpenBMC&#xff1f; OpenBMC 被设计为一个完整的开源 Linux 发行版&#xff0c;可以灵活地进行定制以支持不同的 SoC 或主板。 传统的BMC由固件提供商专有构建&#xff0c;是闭源的&#xff0c;这意味着它不具有在开放平台上修改的灵活性。 另一方面&#xff0c;OpenB…