基于SpringBoot的任务管理三种方式

news2025/1/12 6:08:15

文章目录

  • 前言
  • 一,异步任务
    • 1.1 无返回值异步任务调用
    • 1.2 有返回值异步任务调用
  • 二、定时任务
    • 2.1 背景介绍
    • 2.2 todo
  • 三、邮箱任务
    • 3.1 todo

前言

开发 web 应用时,多数应用都具备任务调度功能,常见的任务包括异步任务、定时任务和邮件任务。我们以数据库报表为例看看任务调度如何帮助改善系统设计。报表可能是错综复杂的,用户可能需要很长时间找到需要的报表数据,此时我们可以在这个报表应用中添加异步任务减少用户等待时间,从而提高用户体验性;除此之外,还可以在报表应用中添加定时任务和邮件任务,以便用户可以安排在任何他们需要的时间定时生成报表,并在 Email 中发送。本文记录如何使用 SpringBoot 开发这些常见的任务。

一,异步任务

web 应用开发中,大多数情况都是通过同步方式完成数据交互处理,但是,当处理与第三方系统的交互时,容易造成响应迟缓的情况,之前大部分都是使用多线程完成此类任务,除此之外,还可以使用异步调用的方式完美解决这个问题。根据异步处理方式的不同,可以将异步任务的调用分为无返回值异步任务调用和有返回值异步任务调用。

1.1 无返回值异步任务调用

  1. 在启动类上添加注解 @EnableAsync(开启基于注解的异步任务支持)
  2. 在指定无返回值的业务方法上添加 @Asyn注解,实现异步请求方法

启动类

@EnableAsync
@SpringBootApplication
public class AsyncApplication {

	public static void main(String[] args) {
		SpringApplication.run(AsyncApplication.class, args);
	}
}

Service 层代码

@Service
public class AsyncService {

    @Async
    public void send(){
        System.out.println("调用短信验证码方法......");
        long start = System.currentTimeMillis();
        try {
            Thread.sleep(4000);
        } catch (InterruptedException e) {
            e.printStackTrace();
            System.out.println("出现异常:"+ e.getMessage());
        }
        long end = System.currentTimeMillis();
        System.out.println("短信验证码方法执行时间:"+(end - start) + "毫秒");
    }
}

Controller 层代码

@RestController
public class AsyncController {
    @Autowired
    private AsyncService asyncService;

    @GetMapping("/async")
    public R testAsync(){
        long start = System.currentTimeMillis();
        asyncService.send();
        long end = System.currentTimeMillis();
        System.out.println("调用结束主流程耗时"+(end-start)+"毫秒");
        return R.ok().put("success", "调用结束主流程耗时"+(end-start)+"毫秒");
    }
}

若没有使用异步处理(注释掉 @Async 注解),访问接口返回数据如下

在这里插入图片描述

控制台输出内容:

在这里插入图片描述

若使用异步处理(使用 @Async 注解),访问接口返回数据如下

在这里插入图片描述

控制台输出内容:

在这里插入图片描述

当未使用异步处理时,访问接口需要等待4s左右才返回数据,而使用异步处理后几乎不需要等待时间,用户体验更佳了。

需要说明的是,无返回值异步方法在被主流程方法调用时,主流程方法不会阻塞,而是继续向下执行主流程方法内容,直接向页面响应结果,而调用的异步方法会作为一个子线程单独执行,直到异步方法执行完成。

1.2 有返回值异步任务调用

  1. 在启动类上添加注解 @EnableAsync(开启基于注解的异步任务支持)
  2. 在指定有返回值的业务方法上添加 @ASync注解,实现异步请求方法
    注意: 返回值必须用Future 泛型封装,例如: new AsyncResult(count),当想获取装的值时用 Futue 对象调用其 get() 方法即可获得。

业务场景设定:假设业务A(耗时3秒)进行数据分析返回一个整数值,业务B(耗时5秒)也进行数据分析返回一个整数值;且两者没有相互影响,计算业务A和业务B的返回整数值相加的结果。

同步方式: 先调用业务A ,再调用B,再将A返回值与B返回值相加,最终将结果返回给用户,总耗时肯定大于8秒。这种方式相对来说整体耗时更长,响应慢,降低了用户体验。

异步方式:

@Async
public Future<Integer> processA(){
    System.out.println("开始分析并统计业务A数据......");
    long start = System.currentTimeMillis();
    try {
        Thread.sleep(3000);
    } catch (InterruptedException e) {
        e.printStackTrace();
        System.out.println("出现异常:"+ e.getMessage());
    }
    Integer numA = 20;
    long end = System.currentTimeMillis();
    System.out.println("业务A数据统计共耗时:"+(end - start));
    return new AsyncResult<Integer>(numA);
}

@Async
public Future<Integer> processB(){
    System.out.println("开始分析并统计业务B数据......");
    long start = System.currentTimeMillis();
    try {
        Thread.sleep(5000);
    } catch (InterruptedException e) {
        e.printStackTrace();
        System.out.println("出现异常:"+ e.getMessage());
    }
    Integer numB = 25;
    long end = System.currentTimeMillis();
    System.out.println("业务B数据统计共耗时:"+(end - start));
    return new AsyncResult<Integer>(numB);
}
@GetMapping("/asyncTask")
public R Async() throws Exception {
    long start = System.currentTimeMillis();
    Future<Integer> processA = asyncService.processA();
    Future<Integer> processB = asyncService.processB();
    int sum = processA.get() + processB.get();
    //等待A B结果相加成功后再向下执行,这里阻塞了
    long end = System.currentTimeMillis();
    System.out.println("调用结束主流程耗时"+(end-start)+"毫秒");
    return R.ok().put("data","success");
}

响应结果:小于8秒,提升了系统响应速度

在这里插入图片描述
需要说明的是,有返回值异步方法在被主流程方法调用时,主流程方法是会短暂阻塞的,需要等待并获取异步方法的返回结果,而调用的两个异步方法会作为两个子线程并行执行,直到异步方法执行完成并返回结果,这样主流程会在最后一个异步方法返回结果后跳出阻塞状态。

二、定时任务

2.1 背景介绍

在实际开发中,可能会有这样一个需求,需要在每天的某个固定时间或者每隔一段时间让程序去执行某一个任务。例如,服务器数据定时在晚上零点备份。通常我们可以使用 scheduling Tasks 实现这一定时任务的处理;在时间定时任务时需要了解下几种和定时任务相关的注解,具体如下:

  1. @EnableScheduling 用于开启基于注解方式的定时任务支持,该注解主要用在项目启动类上。
  2. @scheduled 配置定时任务的执行规则,该注解主要用在定时业务方法上。

@scheduled 注解提供多个属性,精细化配置定时任务执行规则,这些属性及说明如下表:
todo

2.2 todo

三、邮箱任务

3.1 todo

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

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

相关文章

springboot+vue企业固定资产管理系统java

资产管理系统可以更加直观的了解到企业资产的使用情况&#xff0c;让企业资产透明化。资产管理系统可以帮助企业标记企业所有的资产&#xff0c;这些资产包括电脑&#xff0c;桌子&#xff0c;椅子等不动资产的标识&#xff0c;以及固定资产的新增&#xff0c;修改&#xff0c;…

渗透测试 | UserInfo信息收集

0x00 免责声明 本文仅限于学习讨论与技术知识的分享&#xff0c;不得违反当地国家的法律法规。对于传播、利用文章中提供的信息而造成的任何直接或者间接的后果及损失&#xff0c;均由使用者本人负责&#xff0c;本文作者不为此承担任何责任&#xff0c;一旦造成后果请自行承担…

Leetcode 双指针详解

双指针 双指针顾名思义&#xff0c;就是同时使用两个指针&#xff0c;在序列、链表结构上指向的是位置&#xff0c;在树、图结构中指向的是节点&#xff0c;通过或同向移动&#xff0c;或相向移动来维护、统计信息 在数组的区间问题上&#xff0c;暴力算法的时间复杂度往往是O…

分布式项目-规格参数(13)

【今日成果】&#xff1a; //商品维护模块&#xff1b;其中值得一提的是。商品的介绍全部都做成图片的形式&#xff0c;这样有利于去维护。 商品模块中的页面在created中一开始要对会员等级进行查询操作&#xff0c;访问MemberController中的list接口。 //维护规格参数信息…

【离线数仓-9-数据仓库开发DWS层设计要点-1d/nd/td表设计】

离线数仓-9-数据仓库开发DWS层设计要点-1d/nd/td表设计离线数仓-9-数据仓库开发DWS层设计要点-1d/nd/td表设计一、DWS层设计要点二、DWS层设计分析 - 1d/nd1.DWS层设计一&#xff1a;不考虑用户维度2.DWS层设计二&#xff1a;考虑用户维度2.DWS层设计三 &#xff1a;考虑用户商…

骨骼控制(一)——动画动态节点(AnimDynamics)

文章目录一、引言二、骨骼控制三、UE蓝图中提供的骨骼控制节点——AnimDynamics动画蓝图节点1、什么是AnimDynamics动画蓝图节点①使用盒体计算惯性②使用约束来限制移动2、AnimDynamics节点的几种常用例子①单骨骼模拟②骨骼链模拟 <h2 id1>③群魔乱舞&#xff08;这是错…

YOLOv8之C2f模块——与YOLOv5的C3模块对比

一、源码对比 YOLOv8完整工程代码下载&#xff1a;ultralytics/ultralytic   C2f模块源码在ultralytics/nn/modules.py下&#xff0c;源码如下&#xff1a; class C2f(nn.Module):# CSP Bottleneck with 2 convolutionsdef __init__(self, c1, c2, n1, shortcutFalse, g1, e…

安装kibana 报错/访问不了

安装kibana 报错1&#xff0c;elasticsearch.yaml 和kibana.yaml 配置问题2&#xff0c;elasticsearch 和kibana版本不一致3&#xff0c;索引问题1&#xff0c;elasticsearch.yaml 和kibana.yaml 配置问题 我的RPM安装的&#xff0c;配置文件都在/etc/ vim /etc/elasticsearc…

【Python知识点桂电版】02组合数据类型

一、序列序列简介序列是指一种包含多项数据的数据结构&#xff0c;分为不可变序列和可变序列。可变序列可修改序列内的元素如列表&#xff0c;二不可变序列一旦建立就不能修改其中的元素&#xff0c;字符串和元组属于不可变序列。列表和元组的创建列表&#xff1a;列表名 [元素…

大学物理期末大题专题训练总结-热学大题

今天下午去找郑老师权老师等去答疑&#xff0c;老师说大题会考查得比较套路&#xff0c;计算不难。明天就要考试了&#xff0c;再把大题常见题型总结一下&#xff0c;热学这块我做完了蓝本的热学题目&#xff0c;发现了如下三种&#xff1a;有关循环过程曲线的&#xff1a;给出…

简历信息提取论文笔记Information Extraction from Resume Documents in PDF Format

标题&#xff1a;Information Extraction from Resume Documents in PDF Format下载地址&#xff1a;https://library.imaging.org/ei/articles/28/17/art00013长度&#xff1a;8页发表时间&#xff1a;2016引用量cite27先读标题、摘要、结论、然后 methods/experiment design,…

大数据NiFi(二十):实时同步MySQL数据到Hive

文章目录 实时同步MySQL数据到Hive 一、开启MySQL的binlog日志 1、登录mysql查看MySQL是否开启binlog日志 2 、开启mysql binlog日志 3、重启mysql 服务&#xff0c;重新查看binlog日志情况 二、​​​​​​​​​​​​​​配置“CaptureChangeMySQL”处理器 1、创建“…

JAVA企业级开发 1.4 Spring_采用Java配置类管理Bean

文章目录一、采用Java配置类管理Bean&#xff08;一&#xff09;打开项目&#xff08;二&#xff09;创建子包&#xff08;三&#xff09;创建杀龙任务类&#xff08;四&#xff09;创建勇敢骑士类&#xff08;五&#xff09;创建Spring配置类&#xff08;六&#xff09;创建测…

STM32单片机之温湿度检测系统(DTH11、OLED、LCD1602)

LCD1602LCD1602引脚第 1 脚: VSS 为电源地 第 2 脚: VDD 接 5V 正电源 第 3 脚: VL 为液晶显示器对比度调整端,接正电源时对比度最弱&#xff0c;接地时对比度最高&#xff0c;对比度过高时会产生“鬼影”&#xff0c;使用时可以通过一个 10K 的电位器调整对比度。 第 4 脚&…

Windows下命令执行绕过技巧总结(渗透测试专用)

一、连接符1、双引号不要求双引号闭合举例&#xff1a;"who"a"mi" //闭合的 "who"a"mi //不闭合的2、圆括号必须在两边&#xff0c;不能包括中间的字符。举例&#xff1a;((whoami))3、^符号&#xff08;转译符号&#xff09;不可以在结尾&…

Rasa 3.x 学习系列-摆脱意图:一种新的对话模式

Rasa 3.x 学习系列-摆脱意图:一种新的对话模式 在2019年的一篇文章中,Alan Nichol写道 :是时候摆脱意图了。一年后,Rasa发布了Rasa中的第一个无意图(或“端到端”)对话模型。现在,我们宣布迈出了一个重要的步伐,将LLM的强大功能带入Rasa的对话管理中。 首先,意图非常…

YOLOV5s+Shufflenetv2+VOC数据集+迁移学习

前言&#xff1a;更改YOLOV5的backbone网络为 Shufflenetv2&#xff0c;便于达到轻量化的目的 1. 试运行YOLOv5 b站推土机 2. VOC数据集处理 3. 更改轻量级网络 参考魔改yolov5 3.1 在common.py末尾加入以下代码 #添加轻量化模块Shufflenetv2 # ------------------------…

人工智能-机器视觉篇搞定(笔记)

考书目《人工智能之机器视觉》–程晨 1.从计算机读取一张图片显示 ##获取图片 import cv2 imcv2.imread("im.jpg") cv2.imshow("my",im) cv2.waitKey() cv2.destroyAllWindows()2.显示视频帧 import cv2 #cap cv2.VideoCapture("video.mp4")获…

shell脚本入门

实习的时候第一个月的考核就是如何部署一个云资源&#xff0c;当时走的捷径&#xff08;杠杠的搜索能力hhhh&#xff09;找到了一个shell脚本一键部署&#xff0c;后来被leader问起来就如实说了&#xff0c;leader问有没有看懂shell脚本中的逻辑……&#xff08;没有&#xff0…

代码随想录第四十二天| ● 01背包问题,你该了解这些! ● 01背包问题,你该了解这些! 滚动数组 ● 416. 分割等和子集

01背包问题 模型详解 描述 在一个容量有限的 背包里装若干物品&#xff0c;这些物品重量不同&#xff0c;价值不同。如何装使这些背包内物品价值最大。 &#xff08;1&#xff09;如果物品可以分割&#xff0c;直接用贪心算法&#xff0c;首先装价值密度最大的物品 &#xff…