使用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> 更快。这主要有以下几个原因: 优化的底层实现…...
java_网络服务相关_gateway_nacos_feign区别联系
1. spring-cloud-starter-gateway 作用:作为微服务架构的网关,统一入口,处理所有外部请求。 核心能力: 路由转发(基于路径、服务名等)过滤器(鉴权、限流、日志、Header 处理)支持负…...
大数据零基础学习day1之环境准备和大数据初步理解
学习大数据会使用到多台Linux服务器。 一、环境准备 1、VMware 基于VMware构建Linux虚拟机 是大数据从业者或者IT从业者的必备技能之一也是成本低廉的方案 所以VMware虚拟机方案是必须要学习的。 (1)设置网关 打开VMware虚拟机,点击编辑…...
[ICLR 2022]How Much Can CLIP Benefit Vision-and-Language Tasks?
论文网址:pdf 英文是纯手打的!论文原文的summarizing and paraphrasing。可能会出现难以避免的拼写错误和语法错误,若有发现欢迎评论指正!文章偏向于笔记,谨慎食用 目录 1. 心得 2. 论文逐段精读 2.1. Abstract 2…...
反射获取方法和属性
Java反射获取方法 在Java中,反射(Reflection)是一种强大的机制,允许程序在运行时访问和操作类的内部属性和方法。通过反射,可以动态地创建对象、调用方法、改变属性值,这在很多Java框架中如Spring和Hiberna…...
[Java恶补day16] 238.除自身以外数组的乘积
给你一个整数数组 nums,返回 数组 answer ,其中 answer[i] 等于 nums 中除 nums[i] 之外其余各元素的乘积 。 题目数据 保证 数组 nums之中任意元素的全部前缀元素和后缀的乘积都在 32 位 整数范围内。 请 不要使用除法,且在 O(n) 时间复杂度…...
使用Matplotlib创建炫酷的3D散点图:数据可视化的新维度
文章目录 基础实现代码代码解析进阶技巧1. 自定义点的大小和颜色2. 添加图例和样式美化3. 真实数据应用示例实用技巧与注意事项完整示例(带样式)应用场景在数据科学和可视化领域,三维图形能为我们提供更丰富的数据洞察。本文将手把手教你如何使用Python的Matplotlib库创建引…...
C++:多态机制详解
目录 一. 多态的概念 1.静态多态(编译时多态) 二.动态多态的定义及实现 1.多态的构成条件 2.虚函数 3.虚函数的重写/覆盖 4.虚函数重写的一些其他问题 1).协变 2).析构函数的重写 5.override 和 final关键字 1&#…...
为什么要创建 Vue 实例
核心原因:Vue 需要一个「控制中心」来驱动整个应用 你可以把 Vue 实例想象成你应用的**「大脑」或「引擎」。它负责协调模板、数据、逻辑和行为,将它们变成一个活的、可交互的应用**。没有这个实例,你的代码只是一堆静态的 HTML、JavaScript 变量和函数,无法「活」起来。 …...
Python 训练营打卡 Day 47
注意力热力图可视化 在day 46代码的基础上,对比不同卷积层热力图可视化的结果 import torch import torch.nn as nn import torch.optim as optim from torchvision import datasets, transforms from torch.utils.data import DataLoader import matplotlib.pypl…...
Axure 下拉框联动
实现选省、选完省之后选对应省份下的市区...
