SpringBoot 异步编程浅谈

1. 需求背景

  当我们需要提高系统的并发性能时,我们可以将耗时的操作异步执行,从而避免线程阻塞,提高系统的并发性能。例如,在处理大量的并发请求时,如果每个请求都是同步阻塞的方式处

理,系统的响应时间会变得很长。而使用异步编程,可以将一些耗时的操作交给其他线程去处理,从而释放主线程,提高系统的并发能力。

2. SpringBoot如何实现异步调用

  从Spring 3开始,可以通过在方法上标注@Async注解来实现异步方法调用。这意味着当我们调用被@Async注解修饰的方法时,它会在后台以异步方式执行。为了启用异步功能,我们需要

一个配置类,并在该类上使用@EnableAsync注解。这个注解告诉Spring要开启异步功能。

3. 异步调用实现步骤

第一步:新建配置类,开启@Async功能支持

  使用@EnableAsync来开启异步任务支持,@EnableAsync注解可以直接放在SpringBoot启动类上,也可以单独放在其他配置类上。这里选择使用单独的配置类SyncConfiguration

使用@Async注解,在默认情况下用的是SimpleAsyncTaskExecutor线程池,该线程池不是真正意义上的线程池

使用此线程池无法实现线程重用,每次调用都会新建一条线程。若系统中不断的创建线程,最终会导致系统占用内存过高,引发OutOfMemoryError错误,所以在使用Spring中的@Async异步

框架时要自定义线程池,替代默认的SimpleAsyncTaskExecutor,这也是自定义配置的意义之一。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

@Configuration

@EnableAsync

public class SyncConfiguration {

    @Bean(name = "asyncPoolTaskExecutor")

    public ThreadPoolTaskExecutor executor() {

        ThreadPoolTaskExecutor taskExecutor = new ThreadPoolTaskExecutor();

        //核心线程数,设置核心线程数。核心线程数是线程池中一直保持活动的线程数量,即使它们是空闲的。

        taskExecutor.setCorePoolSize(10);

        //设置线程池维护线程的最大数量。当缓冲队列已满并且核心线程数的线程都在忙碌时,线程池会创建新的线程,直到达到最大线程数。

        taskExecutor.setMaxPoolSize(100);

        //设置缓冲队列的容量。当所有的核心线程都在忙碌时,新的任务将会被放入缓冲队列中等待执行。

        taskExecutor.setQueueCapacity(50);

        //设置非核心线程的空闲时间。当超过核心线程数的线程在空闲时间达到设定值后,它们将被销毁,以减少资源的消耗。

        taskExecutor.setKeepAliveSeconds(200);

        //异步方法内部线程名称

        taskExecutor.setThreadNamePrefix("async-");

        /**

         * 当线程池的任务缓存队列已满并且线程池中的线程数目达到maximumPoolSize,如果还有任务到来就会采取任务拒绝策略

         * 通常有以下四种策略:

         * ThreadPoolExecutor.AbortPolicy:丢弃任务并抛出RejectedExecutionException异常。

         * ThreadPoolExecutor.DiscardPolicy:也是丢弃任务,但是不抛出异常。

         * ThreadPoolExecutor.DiscardOldestPolicy:丢弃队列最前面的任务,然后重新尝试执行任务(重复此过程)

         * ThreadPoolExecutor.CallerRunsPolicy:重试添加当前的任务,自动重复调用 execute() 方法,直到成功

         */

        taskExecutor.setRejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy());

        taskExecutor.initialize();

        return taskExecutor;

    }

}

注:

Spring提供了多种线程池:

  • SimpleAsyncTaskExecutor:不是真的线程池,这个类不重用线程,每次调用都会创建一个新的线程。

  • SyncTaskExecutor:这个类没有实现异步调用,只是一个同步操作。只适用于不需要多线程的地

  • ConcurrentTaskExecutor:Executor的适配类,不推荐使用。如果ThreadPoolTaskExecutor不满足要求时,才用考虑使用这个类

  • ThreadPoolTaskScheduler:可以使用cron表达式

  • ThreadPoolTaskExecutor :最常使用,推荐。 其实质是对java.util.concurrent.ThreadPoolExecutor的包装

第二步:在方法上标记异步调用

在异步处理的方法上添加@Async注解,代表该方法为异步处理。

1

2

3

4

5

6

7

8

9

public class AsyncTask {

    @Async

    public void Task() {

        long t1 = System.currentTimeMillis();

        Thread.sleep(5000);

        long t2 = System.currentTimeMillis();

        log.info("task cost {} ms" , t2-t1);

    }

注:

在异步编程中,如果需要处理带有返回值的异步方法(有则继续浏览,无则跳过),Spring提供了java.util.concurrent.Future接口和java.util.concurrent.CompletableFuture类来处理异步任务的返回值。

1. 使用Future接口:Future接口表示一个异步计算的结果。我们可以通过调用Future对象的get()方法来获取异步任务的返回值,但这将阻塞当前线程,直到异步任务完成并返回结果

1

2

3

4

5

6

7

8

9

10

11

12

13

14

@Service

public class MyService {

    @Async

    public Future<String> asyncMethodWithReturnValue() {

        // 模拟耗时操作

        try {

            Thread.sleep(2000);

        catch (InterruptedException e) {

            e.printStackTrace();

        }

         

        return new AsyncResult<>("Async method result");

    }

}

上述示例中,asyncMethodWithReturnValue()方法使用@Async注解标记为异步方法,并返回一个Future<String>对象。在方法内部,我们使用AsyncResult类来创建一个包含异步结果的Future对象。

在调用该异步方法时,可以使用get()方法来获取异步任务的返回值。但需要注意,get()方法会阻塞当前线程,直到异步任务执行完成并返回结果。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

@Autowired

private MyService myService;

public void someMethod() {

    Future<String> futureResult = myService.asyncMethodWithReturnValue();

     

    // 阻塞等待异步任务完成并获取返回值

    try {

        String result = futureResult.get();

        System.out.println("Async method result: " + result);

    catch (InterruptedException | ExecutionException e) {

        e.printStackTrace();

    }

}

上述示例中,我们通过调用futureResult.get()方法来获取异步任务的返回值。如果异步任务还未完成,调用get()方法将会阻塞当前线程,直到异步任务完成并返回结果。

2. 使用CompletableFuture类:CompletableFuture是Java 8引入的一个强大的异步编程工具,提供了更加灵活和功能丰富的异步任务处理方式。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

@Service

public class MyService {

    @Async

    public CompletableFuture<String> asyncMethodWithReturnValue() {

        // 模拟耗时操作

        try {

            Thread.sleep(2000);

        catch (InterruptedException e) {

            e.printStackTrace();

        }

         

        return CompletableFuture.completedFuture("Async method result");

    }

}

在上述示例中,asyncMethodWithReturnValue()方法使用@Async注解标记为异步方法,并返回一个CompletableFuture<String>对象。在方法内部,我们使用CompletableFuture.completedFuture()方法创建一个包含异步结果的CompletableFuture对象。

在调用该异步方法时,可以链式调用thenApply()thenAccept()等方法来对异步任务的结果进行处理。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

@Autowired

private MyService myService;

public void someMethod() {

    CompletableFuture<String> futureResult = myService.asyncMethodWithReturnValue();

     

    futureResult.thenAccept(result -> {

        System.out.println("Async method result: " + result);

    });

     

    // 执行其他操作

     

    // 阻塞等待异步任务完成

    futureResult.join();

}

在上述示例中,我们通过调用thenAccept()方法来处理异步任务的结果,而不需要显式调用get()方法。thenAccept()方法接受一个Consumer函数式接口,用于处理异步任务的结果。

此外,CompletableFuture还提供了丰富的方法,例如thenApplyAsync()thenCompose()thenCombine()等,用于处理复杂的异步任务流程。

注:

当使用CompletableFuture处理异步任务时,以下是thenApplyAsync()thenCompose()thenCombine()thenAccept()这四个方法的区别

  • thenApplyAsync()thenAccept()用于处理异步任务的结果,并返回一个新的CompletableFuture或不返回任何结果。
  • thenCompose()用于处理异步任务的结果,并返回一个新的CompletableFuture,该结果是另一个CompletionStage的结果。
  • thenCombine()用于组合两个异步任务的结果,并应用指定的函数处理结果,并返回一个新的CompletableFuture
第三步:在需要进行异步执行的地方进行调用

1

asyncTask.Task();

4. @Async的原理

  1. 当一个带有@Async注解的方法被调用时,Spring会创建一个异步代理对象来代理这个方法的调用。

  2. 异步代理对象会将方法调用封装为一个独立的任务,并将该任务提交给异步任务执行器。

  3. 异步任务执行器从线程池中获取一个空闲的线程,并将任务分配给该线程执行。

  4. 调用线程立即返回,不会等待异步任务的执行完成。

  5. 异步任务在独立的线程中执行,直到任务完成。

  6. 异步任务执行完成后,可以选择返回结果或者不返回任何结果。

最后感谢每一个认真阅读我文章的人,礼尚往来总是要有的,虽然不是什么很值钱的东西,如果你用得到的话可以直接拿走:

这些资料,对于【软件测试】的朋友来说应该是最全面最完整的备战仓库,这个仓库也陪伴上万个测试工程师们走过最艰难的路程,希望也能帮助到你!

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

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

相关文章

Git使用教程 gittutorial

该教程对该文章的翻译&#xff1a;https://git-scm.com/docs/gittutorial 本文介绍怎用使用 Git 导入新的工程、修改文件及如何其他人同步开发。 首先&#xff0c; 可以使用以下指令获取文档帮助 git help log笔者注&#xff1a;不建议看这个文档&#xff0c;标准的语法介绍…

FreeRTOS的学习

1.创建函数和删除 动态创建为FreeRTOS分配的堆栈&#xff08;方便&#xff09;&#xff0c;而静态创建为人为分配空间。动态应用多任务中必须有while&#xff08;1&#xff09;否则只会执行一次任务中的延时要用 vTaskDelay(500); 延时期间执行其它任务 任务中的延时使…

postman进阶使用

前言 对于postman的基础其实很容易上手实现&#xff0c;也有很多教程。 对于小编我来说&#xff0c;也基本可以实现开发任务。 但是今年我们的高级测试&#xff0c;搞了一下postman&#xff0c;省去很多工作&#xff0c;让我感觉很有必要学一下 这篇文章是在 高级测试工程师ht…

01-Spring Security框架的认证和授权测试

Spring Security 介绍 认证功能与业务无关几乎是每个项目都要具备的功能,市面上有很多认证框架如Apache Shiro、CAS、Spring Security等 Spring Security是Spring家族的一份子且和Spring Cloud集成的很好&#xff0c;所以本项目采用Spring Security作为认证服务的技术框架 …

vue3(六)-基础入门之自定义组件与插槽、ref通信

一、全局组件 html: <div id"app"><mytemplace></mytemplace> </div>javascript: <script>const { createApp } Vueconst app createApp({})app.component(mytemplace, {template: <div><button>返回</button>…

canvas随机绘制100个五角星

canvas实例应用100 专栏提供canvas的基础知识&#xff0c;高级动画&#xff0c;相关应用扩展等信息。 canvas作为html的一部分&#xff0c;是图像图标地图可视化的一个重要的基础&#xff0c;学好了canvas&#xff0c;在其他的一些应用上将会起到非常重要的帮助。 文章目录 示例…

element-plus修改主题颜色

一、自定义scss文件 在src\css\styles\element目录下新建index.scss 代码如下 forward "element-plus/theme-chalk/src/common/var.scss" with ($colors: ("primary": ("base": #d61b1a,"color": #fff,),) );use "element-plus…

Java - 工厂设计模式

Java - 工厂设计模式 一. 简介二. 例子2.1 定义抽象类2.2 定义子类2.3 创建工厂2.4 测试 三. JDK中使用工厂模式的案例 前言 这是我在这个网站整理的笔记,有错误的地方请指出&#xff0c;关注我&#xff0c;接下来还会持续更新。 作者&#xff1a;神的孩子都在歌唱 工厂设计模式…

Day73力扣打卡

打卡记录 统计移除递增子数组的数目 II&#xff08;双指针&#xff09; 链接 class Solution:def incremovableSubarrayCount(self, a: List[int]) -> int:n len(a)i 0while i < n - 1 and a[i] < a[i 1]:i 1if i n - 1: # 每个非空子数组都可以移除return n …

普中STM32-PZ6806L开发板(STM32CubeMX创建项目并点亮LED灯)

简介 搭建一个用于驱动 STM32F103ZET6 GPIO点亮LED灯的任务;电路原理图 LED电路原理图 芯片引脚连接LED驱动引脚原理图 创建一个点亮LED灯的Keil 5项目 创建STM32CubeMX项目 New Project -> 单击 -> 芯片搜索STM32F103ZET6->双击创建 初始化时钟 调试设置 一…

双侧电源系统距离保护MATLAB仿真模型

微❤关注“电气仔推送”获得资料&#xff08;专享优惠&#xff09; 系统原始数据 双侧电源系统模型如图所示&#xff1a; 仿真模型搭建 将线路AB分成Line1和Line2&#xff0c;将线路BC分成Line3和Line4&#xff0c;用三相电压电流测量模块作为系统母线&#xff0c;根据系统已…

洛谷 P1387 最大正方形 刷题笔记

P1387 最大正方形 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn) 找出一个 由数字1组成的最大正方形 输出该正方形的边长 &#xff1b; 思路 dp 画图模拟 可以发现 当 当前点a[i][j]1,满足构成正方形的初步条件 而当前点能构成的最大正方形长度 由它的左上角 左边和上边…

新手为什么跟着大型机构交易?fpmarkets总结理由

正所谓方向不对努力白费&#xff0c;这也就是为什么fpmarkets建议新手在刚开始的时候&#xff0c;跟着大型机构进行交易。 这些大型机构包括中央银行、巨额对冲基金、投资和保险公司等等&#xff0c;首先fpmarkets认为这些大型机构的交易量巨大&#xff0c;能够影响市场的走势。…

LeetCode刷题--- N 皇后

个人主页&#xff1a;元清加油_【C】,【C语言】,【数据结构与算法】-CSDN博客 个人专栏 力扣递归算法题 http://t.csdnimg.cn/yUl2I 【C】 http://t.csdnimg.cn/6AbpV 数据结构与算法 ​​​​​​http://t.csdnimg.cn/hKh2l 前言&#xff1a;这个专栏主要讲述递归…

ARCGIS PRO SDK 要素空间关系

一、要素与要素查询&#xff0c;返回的是bool值 1、 Touches 判断几何要素是否接触 Touches 如果 geometry1 与 geometry2 接触&#xff0c;则返回 true&#xff0c;否则 false。 touches GeometryEngine.Instance.Touches(Geometry1, Geometry2) 2、…

传感器原理与应用复习--电感式传感器

文章目录 上一篇自感式电感传感器差动变压器式传感器电涡流式传感器下一篇 上一篇 传感器原理与应用–传感器基本特性与应变式传感器 自感式电感传感器 将线圈通入电流产生磁场&#xff0c;而间隙的大小将影响磁场的大小&#xff0c;从而进行检测 变气隙式电感传感器&#xf…

2D transform 1-translate

移位&#xff1a;translate 缩放&#xff1a;scale 旋转&#xff1a;rotate 扭曲&#xff1a;skew <style>.outer {width: 200px;height: 200px;border: 2px solid black;margin-top: 100px;}.inner {width: 200px;height: 200px;background-color: pink;transform: t…

el-select多选修改默认显示一个

需求&#xff1a;超出选择框已数字展示 效果 备忘一下 代码 父组件 <template><div><div class"credit_box"><div class"credit_select_box"><div class"credit_select"><span>选择框1</span><…

C#中的Attribute详解(上)

C#中的Attribute详解&#xff08;上&#xff09; 一、Attribute是什么二、Attribute的作用三、Attribute与注释的区别四、系统Attribute范例1、如果不使用Attribute&#xff0c;为了区分这四类静态方法&#xff0c;我们只能通过注释来说明&#xff0c;但这样做会给系统带来很多…

前端文件在虚拟机,后端在本机,两个如何通信

前端文件在虚拟机&#xff0c;后端在本机&#xff0c;两个如何通信 如果前端的文件放在虚拟机里面&#xff0c;但是调用接口的后端在本地调试&#xff0c;如何做到在虚拟机中也能访问到本地的接口内容。 其实这个问题很简单&#xff0c;只要讲本地的IP和虚拟机中的IP结合就可…