使用ExecutorService和@Async来使用多线程
文章目录
- 使用ExecutorService和@Async来使用多线程
- 采用ExecutorService来使用多线程
- 多线程过程的详细解释
- 注意事项
- 优点
- 使用@Async来使用多线程
- 对比@Async和ExecutorService的多线程使用方式
- 使用 `ExecutorService` 的服务类
- 使用 `@Async` 的服务类
- 异步任务类
- 自定义线程池
- 主应用类
- 解释
- 运行这个示例
- 注意点:
使用ExecutorService和@Async来使用多线程
采用ExecutorService来使用多线程
我们直接来通过一段代码来看一下多线程的过程,大致就是通过多线程来生成图片
@Service
public class NameService {private final ExecutorService executorService = Executors.newFixedThreadPool(10); // 创建一个固定大小的线程池public void createImg(String fileNamePrefix) {// 假设我们有一组需要生成的图片List<String> imagePrefixes = getImagePrefixes(fileNamePrefix);try {for (String prefix : imagePrefixes) {// 提交任务executorService.submit(() -> {generateImage(prefix); // 生成图片的方法});}} catch(Exception e) {e.printStackTrace();} finally {// 关闭线程池,但允许已提交的任务继续执行executorService.shutdown();// 下面是为了确保线程池关闭,手动设置一个时间,如果线程60秒没有执行完,则强制关闭该线程,我这里不需要// try {// if (!executorService.awaitTermination(60, TimeUnit.SECONDS)) {// executorService.shutdownNow();// }// } catch (InterruptedException e) {// executorService.shutdownNow();// Thread.currentThread().interrupt();// }}}// 模拟要生成的图片名列表private List<String> getImagePrefixes(String fileNamePrefix) {// 获取需要生成的图片前缀列表,示例代码return Arrays.asList(fileNamePrefix + "_1", fileNamePrefix + "_2", fileNamePrefix + "_3");}// 生成图片的方法private void generateImage(String prefix) {// 生成图片并保存到本地的方法,示例代码System.out.println("Generating image for prefix: " + prefix);// 实际的图片生成逻辑,这里简单用一个延时函数代替try { Thread.sleep(1000); // 模拟生成图片的时间 } catch (InterruptedException e) { Thread.currentThread().interrupt(); } System.out.println("Generating image for prefix: " +}
}// 注释相关解释:
//awaitTermination:等待所有任务在指定时间内完成。如果在指定时间(60 秒)内所有任务没有完成,则调用 executorService.shutdownNow() 强制关闭线程池,并中断所有正在执行的任务。
//异常处理:处理可能的 InterruptedException,确保线程池能够被正确关闭。
多线程过程的详细解释
- 创建线程池: 使用
Executors.newFixedThreadPool(10)创建一个固定大小为 10 的线程池。你可以根据需要调整线程池的大小。 - 提交任务: 使用
executorService.submit()方法将图片生成任务提交给线程池,这些任务会被放入线程池的任务队列中,等待线程池中的线程来执行。 - 执行任务: 线程池中的线程会从任务队列中取出任务并执行。使用的是
ExecutorService和多线程,即使for循环结束,提交的任务仍会在后台继续运行,直到所有任务完成。 - 关闭线程池: 调用
executorService.shutdown()方法后,线程池不再接受新任务,但会继续执行已经提交的任务,直到所有任务执行完毕。如果generateImage方法耗时较长,那么这些任务会在后台继续运行,直到所有任务完成。
多线程使得任务的提交与执行分离。for 循环快速提交任务,而线程池在后台并发地执行这些任务。这种异步执行的机制允许你同时处理多个任务,而不需要等待每个任务的完成。
注意事项
- 任务数量:确保线程池大小与任务数量合理匹配,以免线程过多或过少影响性能。
- 异常处理:可以在任务中添加异常处理逻辑,确保不会因为某个任务失败而影响其他任务的执行。
优点
- 提高并发性:多个任务可以同时进行,提高效率。
- 主线程不中断:主线程可以继续执行其他操作,而不需要等待每个任务的完成。
使用@Async来使用多线程
Spring 提供了 @Async 注解,用于简化多线程编程。通过这个注解让 Spring 自动管理线程池。
启用异步支持: 首先,在 Spring Boot 应用中启用异步支持。可以在主应用类或配置类上添加 @EnableAsync 注解。
标记异步方法: 在需要异步执行的方法上添加 @Async 注解。Spring 会在后台线程池中执行这些方法,不会阻塞主线程。
具体例子咱们可以通过下面的对比来看。
对比@Async和ExecutorService的多线程使用方式
使用 ExecutorService 的服务类
package caj.springboot.service;import org.springframework.scheduling.annotation.EnableAsync;
import org.springframework.stereotype.Service;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.List;
import java.util.Arrays;/*** @description: 使用 `ExecutorService` 的服务类* @author: CAJ* @time: 2025/1/2 20:13*/
@Service
public class ExecutorServiceNameService {private final ExecutorService executorService = Executors.newFixedThreadPool(10); // 创建一个固定大小的线程池public void createImg(String fileNamePrefix) {List<String> imagePrefixes = getImagePrefixes(fileNamePrefix);long startTime = System.currentTimeMillis();for (String prefix : imagePrefixes) {executorService.submit(() -> {generateImage(prefix); // 生成图片的方法});}executorService.shutdown();// 用来判断线程池中的所有任务是否都已经完成。while (!executorService.isTerminated()) {// 等待所有任务完成}long endTime = System.currentTimeMillis();System.out.println("ExecutorService 总耗时: " + (endTime - startTime) + " 毫秒");}private List<String> getImagePrefixes(String fileNamePrefix) {return Arrays.asList(fileNamePrefix + "_1", fileNamePrefix + "_2", fileNamePrefix + "_3");}private void generateImage(String prefix) {try {Thread.sleep(1000); // 模拟生成图片的时间} catch (InterruptedException e) {Thread.currentThread().interrupt();}System.out.println("Generating image for prefix: " + prefix);}
}
使用 @Async 的服务类
(之后将这个异步方法抽出来了,这里面只是调用,否则无法异步执行)
package caj.springboot.service;import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.scheduling.annotation.Async;
import org.springframework.stereotype.Service;
import java.util.List;
import java.util.Arrays;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.ExecutionException;
import java.util.stream.Collectors;/*** @description:* @author: CAJ* @time: 2025/1/2 20:13*/
@Service
public class AsyncNameService {@Autowiredprivate AsyncTask asyncTask;public void createImg(String fileNamePrefix) {List<String> imagePrefixes = getImagePrefixes(fileNamePrefix);long startTime = System.currentTimeMillis();// 下面这样写只是为了确保整个异步过程结束才会记录endTime,正常使用就for循环然后调用方法即可List<CompletableFuture<Void>> futures = imagePrefixes.stream().map(prefix -> asyncTask.generateImageAsync(prefix)).collect(Collectors.toList());// 等待所有异步任务完成CompletableFuture<Void> allOf = CompletableFuture.allOf(futures.toArray(new CompletableFuture[0]));try {allOf.get(); // 阻塞等待所有任务完成} catch (InterruptedException | ExecutionException e) {e.printStackTrace();}long endTime = System.currentTimeMillis();System.out.println("Async 总耗时: " + (endTime - startTime) + " 毫秒");}private List<String> getImagePrefixes(String fileNamePrefix) {return Arrays.asList(fileNamePrefix + "_1", fileNamePrefix + "_2", fileNamePrefix + "_3");}// 本来将异步方法直接在这个类中写,但是发现无论如何都无法异步执行,顾后面将异步方法抽出来
// /**
// * 在异步方法 generateImageAsync 中返回一个 CompletableFuture,表示异步任务的完成。
// * @param prefix
// * @return
// */
// @Async("taskExecutor")
// public CompletableFuture<Void> generateImageAsync(String prefix) {
// System.out.println(Thread.currentThread().getName() + " is processing " + prefix);
// generateImage(prefix);
// return CompletableFuture.completedFuture(null);
// }
//
//
// private void generateImage(String prefix) {
// try {
// Thread.sleep(1000); // 模拟生成图片的时间
// } catch (InterruptedException e) {
// Thread.currentThread().interrupt();
// }
// System.out.println("Generating image for prefix: " + prefix);
// }
}
异步任务类
package caj.springboot.service;import org.springframework.scheduling.annotation.Async;
import org.springframework.stereotype.Component;
import java.util.concurrent.CompletableFuture;/*** @description: 异步任务类* @author: CAJ* @time: 2025/1/2 20:52*/
@Component
public class AsyncTask {@Async("taskExecutor")public CompletableFuture<Void> generateImageAsync(String prefix) {generateImage(prefix);return CompletableFuture.completedFuture(null);}private void generateImage(String prefix) {try {Thread.sleep(1000); // 模拟生成图片的时间} catch (InterruptedException e) {Thread.currentThread().interrupt();}System.out.println("Generating image for prefix: " + prefix);}
}
自定义线程池
@Configuration
@EnableAsync
public class AsyncConfig {@Bean(name = "taskExecutor")public Executor taskExecutor() {ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();executor.setCorePoolSize(10);executor.setMaxPoolSize(20);executor.setQueueCapacity(500);executor.setThreadNamePrefix("Async-");executor.initialize();return executor;}
}
- 自定义线程池配置类:通过实现
AsyncConfigurer接口和定义getAsyncExecutor方法,配置自定义的线程池。 - 自定义线程池参数:设置核心线程数、最大线程数、队列容量和线程名前缀。
使用 @Async 注解和自定义线程池,你可以轻松地将方法异步执行,从而提高并发处理能力并优化应用程序性能。
主应用类
import org.springframework.boot.CommandLineRunner;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.context.annotation.Bean;@SpringBootApplication
public class AsyncDemoApplication {public static void main(String[] args) {SpringApplication.run(AsyncDemoApplication.class, args);}@BeanCommandLineRunner run(ExecutorServiceNameService executorServiceNameService, AsyncNameService asyncNameService) {return args -> {System.out.println("使用 ExecutorService:");executorServiceNameService.createImg("test");System.out.println("使用 @Async:");asyncNameService.createImg("test");};}
}
运行结果:
使用 ExecutorService:
Generating image for prefix: test_2
Generating image for prefix: test_1
Generating image for prefix: test_3
ExecutorService 总耗时: 1003 毫秒
使用 @Async:
Generating image for prefix: test_2
Generating image for prefix: test_1
Generating image for prefix: test_3
Async 总耗时: 1009 毫秒
由于这里的例子设置时间比较短,所以最终时间差不多
解释
- ExecutorServiceNameService:使用
ExecutorService处理图片生成任务,创建一个固定大小为 10 的线程池。 - AsyncNameService:使用
@Async注解让方法异步执行,依赖 Spring 管理的线程池。 - 主应用类:在应用启动时,运行两个服务并输出各自的总耗时,以便对比。
运行这个示例
- 将这两个服务类和主应用类放入项目中,运行主应用程序。
- 程序会分别调用
ExecutorServiceNameService和AsyncNameService的createImg方法,并输出各自的总耗时。
通过这个对比,你可以清楚地看到使用 ExecutorService 和 @Async 的不同之处以及它们的性能表现。
注意点:
- 为了确保
@Async注解生效,可以将异步方法放在一个独立的类中,并从主服务类调用这个异步方法。这样,Spring AOP 代理能够正确拦截方法调用并应用异步特性,因为Spring 使用 AOP 代理来拦截方法调用并提供额外的功能,比如事务管理、异步执行等, - 同一个类内部自调用:当你在同一个类内部调用带有
@Async注解的方法时,Spring AOP 代理无法拦截这个调用,从而无法应用异步特性。这是因为代理对象拦截方法调用的前提是调用发生在代理对象本身上,而不是在同一个类内部。
相关文章:
使用ExecutorService和@Async来使用多线程
文章目录 使用ExecutorService和Async来使用多线程采用ExecutorService来使用多线程多线程过程的详细解释注意事项优点 使用Async来使用多线程对比Async和ExecutorService的多线程使用方式使用 ExecutorService 的服务类使用 Async 的服务类异步任务类自定义线程池主应用类解释…...
计算机网络 (19)扩展的以太网
前言 以太网(Ethernet)是一种局域网(LAN)技术,它规定了包括物理层的连线、电子信号和介质访问层协议的内容。以太网技术不断演进,从最初的10Mbps到如今的10Gbps、25Gbps、40Gbps、100Gbps等,已成…...
构造器/构造方法
1. 构造器 1.1 概述 先浏览下面简单代码; class Cons{ // 属性int age;String name; // 方法public void show(){System.out.println("age"age);} } class ConsTest{public static void main(String[] args) {Cons c new Cons();// Cons() 就是…...
异常
目录 1. 异常的概念及使用 1.1 异常的概念 1.2 异常的抛出和捕获 1.3 栈展开 1.4 查找匹配的处理代码 1.5 异常的重新抛出 1.6 异常安全问题 1.7 异常规范 2. 标准库的异常 1. 异常的概念及使用 1.1 异常的概念 异常处理机制允许程序中独⽴开发的部分能够在运⾏时就…...
MySQL中distinct和group by去重的区别
MySQL中distinct和group by去重的区别 在MySQL中,我们经常需要对查询结果进行去重,而DISTINCT和GROUP BY是实现这一功能的两种常见方法。虽然它们在很多情况下可以互换使用,但它们之间还是存在一些差异的。接下来,我们将通过创建测…...
Qt判别不同平台操作系统调用相应动态库读取RFID
本示例使用的读卡器:https://item.taobao.com/item.htm?spma21dvs.23580594.0.0.52de2c1b8jdyXi&ftt&id562957272162 #include <QDebug> #include "mainwindow.h" #include "./ui_mainwindow.h" #include "QLibrary"…...
vue2+echarts实现水球+外层动效
实现效果 安装echarts-liquidfill 需要安装echarts-liquidfill!!!需要安装echarts-liquidfill!!!需要安装echarts-liquidfill!!! 安装命令 npm install echarts-liqui…...
C++ 基础思维导图(一)
目录 1、C基础 IO流 namespace 引用、const inline、函数参数 重载 2、类和对象 类举例 3、 内存管理 new/delete 对象内存分布 内存泄漏 4、继承 继承权限 继承中的构造与析构 菱形继承 1、C基础 IO流 #include <iostream> #include <iomanip> //…...
【gopher的java学习笔记】依赖管理方式对比(go mod maven)
什么是go mod go mod是Go语言官方引入的模块管理工具,旨在简化项目依赖管理,提高构建的可重复性和稳定性。以下是关于go mod的详细介绍: 在go mod之前,Go语言主要依赖GOPATH和vendor目录来管理项目依赖。然而,这种方式…...
CTFshow—远程命令执行
29-35 Web29 代码利用正则匹配过滤了flag,后面加了/i所以不区分大小写。 可以利用通配符绕过 匹配任何字符串/文本,包括空字符串;*代表任意字符(0个或多个) ls file * ? 匹配任何一个字符(不…...
Qt之简易音视频播放器设计(十五)
Qt开发 系列文章 - MediaPlayer(十五) 目录 前言 一、QMediaPlayer 二、实现方式 1.添加multimedia 2.创建类vedioplayer 3.UI设计 4.用户使用 5.效果演示 总结 前言 利用Qt进行音视频播放器设计,首先比较方便使用的是Qt自带的音视…...
ArrayList 和LinkedList的区别比较
前言 ArrayList和LinkedList的主要区别在于它们的底层数据结构、性能特点以及适用场景。ArrayList和LinkedList从名字分析,他们一个是Array(动态数组)的数据结构,一个是Linked(链表)的数据结构&#x…...
Wallpaper壁纸制作学习记录13
骨骼物理模拟 Wallpaper Engine还允许您为人偶变形骨骼配置某些物理模拟。选择骨骼时,点击编辑约束来配置骨骼这些属性。 警告 请记住,物理模拟可能会根据用户的最大FPS设置略微改变其行为。 Wallpaper Engine编辑器将始终以高帧速率渲染。您可以将壁纸…...
Visual Studio 2022安装教程
1、下载网址 Visual Studio 2022 IDE安装网址借助 Visual Studio 设计,具有自动完成、构建、调试、测试功能的代码将与 Git 管理和云部署融为一体。https://visualstudio.microsoft.com/zh-hans/vs/ 点击图片所示 双击运行 2、安装 点击C桌面开发(右边…...
std__invoke 的使用
std__invoke 的使用 文章目录 std__invoke 的使用1. std::invoke 的功能2. 语法3. 使用场景1. 调用普通函数2. 调用成员函数3. 调用成员函数(通过指针或引用)4. 调用函数对象(仿函数)5. 调用 Lambda 表达式 4. std::invoke 的优势…...
2501d,d.109
原文 2.109.0带来了15个主要更改和26个修复的Bugzilla问题.非常感谢39位贡献者,是他们使2.109.0变成可能. 更改编译器 1,[下一版]现在,为类型实例的成员设置别名是个错误 2,添加位字段内省功能 3,添加了从CTFE写入消息的__ctfeWrite 4,现在-verrors也限制弃用警告 5,dtoh为e…...
1、蓝牙打印机环境搭建
本项目采用stm32f103c8T6芯片,通过库函数实现打印功能,并配置有小程序蓝牙通信上位机。 1、创建文件夹目录 core文件夹存放核心库文件 LIB文件夹存放标准库函数文件 这里可以删减,用不到的可以不要。 obj存放编译后的文件 project存放项目…...
Axure RP11安装学习
安装: 官网下载地址:Axure RP - UX Prototypes, Specifications, and Diagrams in One Tool 设置自己的安装目录,一步步安装即可。 汉化: 汉化包下载地址: 链接: https://pan.baidu.com/s/1eIRoGkVqAY3u3I27lgDJ6A…...
axios和fetch的实现原理以及区别,与XMLHttpRequest的关系,并结合react封装统一请求示例
Axios 和 Fetch 对比及统一请求封装 1. Axios 基础用法 1.1 安装和引入 // 安装 npm install axios// 引入 import axios from axios;1.2 基本请求方法 // GET 请求 axios.get(/api/users).then(response > console.log(response.data)).catch(error > console.error…...
矩阵运算提速——玩转opencv::Mat
介绍:用Eigen或opencv::Mat进行矩阵的运算,比用cpp的vector或vector进行矩阵运算要快吗? 使用 Eigen 或 OpenCV 的 cv::Mat 进行矩阵运算通常比使用 std::vector<int> 或 std::vector<double> 更快。这主要有以下几个原因: 优化的底层实现…...
多云管理“拦路虎”:深入解析网络互联、身份同步与成本可视化的技术复杂度
一、引言:多云环境的技术复杂性本质 企业采用多云策略已从技术选型升维至生存刚需。当业务系统分散部署在多个云平台时,基础设施的技术债呈现指数级积累。网络连接、身份认证、成本管理这三大核心挑战相互嵌套:跨云网络构建数据…...
生成xcframework
打包 XCFramework 的方法 XCFramework 是苹果推出的一种多平台二进制分发格式,可以包含多个架构和平台的代码。打包 XCFramework 通常用于分发库或框架。 使用 Xcode 命令行工具打包 通过 xcodebuild 命令可以打包 XCFramework。确保项目已经配置好需要支持的平台…...
手游刚开服就被攻击怎么办?如何防御DDoS?
开服初期是手游最脆弱的阶段,极易成为DDoS攻击的目标。一旦遭遇攻击,可能导致服务器瘫痪、玩家流失,甚至造成巨大经济损失。本文为开发者提供一套简洁有效的应急与防御方案,帮助快速应对并构建长期防护体系。 一、遭遇攻击的紧急应…...
使用VSCode开发Django指南
使用VSCode开发Django指南 一、概述 Django 是一个高级 Python 框架,专为快速、安全和可扩展的 Web 开发而设计。Django 包含对 URL 路由、页面模板和数据处理的丰富支持。 本文将创建一个简单的 Django 应用,其中包含三个使用通用基本模板的页面。在此…...
日语学习-日语知识点小记-构建基础-JLPT-N4阶段(33):にする
日语学习-日语知识点小记-构建基础-JLPT-N4阶段(33):にする 1、前言(1)情况说明(2)工程师的信仰2、知识点(1) にする1,接续:名词+にする2,接续:疑问词+にする3,(A)は(B)にする。(2)復習:(1)复习句子(2)ために & ように(3)そう(4)にする3、…...
【入坑系列】TiDB 强制索引在不同库下不生效问题
文章目录 背景SQL 优化情况线上SQL运行情况分析怀疑1:执行计划绑定问题?尝试:SHOW WARNINGS 查看警告探索 TiDB 的 USE_INDEX 写法Hint 不生效问题排查解决参考背景 项目中使用 TiDB 数据库,并对 SQL 进行优化了,添加了强制索引。 UAT 环境已经生效,但 PROD 环境强制索…...
QMC5883L的驱动
简介 本篇文章的代码已经上传到了github上面,开源代码 作为一个电子罗盘模块,我们可以通过I2C从中获取偏航角yaw,相对于六轴陀螺仪的yaw,qmc5883l几乎不会零飘并且成本较低。 参考资料 QMC5883L磁场传感器驱动 QMC5883L磁力计…...
srs linux
下载编译运行 git clone https:///ossrs/srs.git ./configure --h265on make 编译完成后即可启动SRS # 启动 ./objs/srs -c conf/srs.conf # 查看日志 tail -n 30 -f ./objs/srs.log 开放端口 默认RTMP接收推流端口是1935,SRS管理页面端口是8080,可…...
第一篇:Agent2Agent (A2A) 协议——协作式人工智能的黎明
AI 领域的快速发展正在催生一个新时代,智能代理(agents)不再是孤立的个体,而是能够像一个数字团队一样协作。然而,当前 AI 生态系统的碎片化阻碍了这一愿景的实现,导致了“AI 巴别塔问题”——不同代理之间…...
C++ 求圆面积的程序(Program to find area of a circle)
给定半径r,求圆的面积。圆的面积应精确到小数点后5位。 例子: 输入:r 5 输出:78.53982 解释:由于面积 PI * r * r 3.14159265358979323846 * 5 * 5 78.53982,因为我们只保留小数点后 5 位数字。 输…...
