2021-09-14 Springboot中使用@Async实现异步调用

同步调用:程序按照定义顺序依次执行,每一行程序都必须等待上一行程序执行完成之后才能执行;

异步调用:程序在顺序执行时,不等待异步调用的语句返回结果就执行后面的程序。

同步调用

  1. 编写任务类,创建三个处理函数分别模拟三个执行任务的操作,操作消耗时间随机取(10秒内)
@Slf4j
@Component
public class AsyncTasks {
    public static Random random = new Random();

    public void doTaskOne() throws Exception {
        log.info("开始做任务一");
        long start = System.currentTimeMillis();
        Thread.sleep(random.nextInt(10000));
        long end = System.currentTimeMillis();
        log.info("完成任务一,耗时:" + (end - start) + "毫秒");
    }

    public void doTaskTwo() throws Exception {
        log.info("开始做任务二");
        long start = System.currentTimeMillis();
        Thread.sleep(random.nextInt(10000));
        long end = System.currentTimeMillis();
        log.info("完成任务二,耗时:" + (end - start) + "毫秒");
    }

    public void doTaskThree() throws Exception {
        log.info("开始做任务三");
        long start = System.currentTimeMillis();
        Thread.sleep(random.nextInt(10000));
        long end = System.currentTimeMillis();
        log.info("完成任务三,耗时:" + (end - start) + "毫秒");
    }
}
  1. 在测试类中调用
@Slf4j
@SpringBootTest
public class Test01ApplicationTests {

    @Autowired
    private AsyncTasks asyncTasks;

    @Test
    public void test() throws Exception {
        asyncTasks.doTaskOne();
        asyncTasks.doTaskTwo();
        asyncTasks.doTaskThree();
    }

}

异步调用

同步调用执行时间比较长,若这三个任务本身之间不存在依赖关系,可以并发执行,同步调用执行效率比较差,可以考虑通过异步调用的方式来并发执行。

在Springboot中,可以使用@Async注解就能简单的将原来的同步函数变为异步函数,Task类改在为如下模式:

@Slf4j
@Component
public class AsyncTasks {
    public static Random random = new Random();

    @Async
    public void doTaskOne() throws Exception {
        log.info("开始做任务一");
        long start = System.currentTimeMillis();
        Thread.sleep(random.nextInt(10000));
        long end = System.currentTimeMillis();
        log.info("完成任务一,耗时:" + (end - start) + "毫秒");
    }

    @Async
    public void doTaskTwo() throws Exception {
        log.info("开始做任务二");
        long start = System.currentTimeMillis();
        Thread.sleep(random.nextInt(10000));
        long end = System.currentTimeMillis();
        log.info("完成任务二,耗时:" + (end - start) + "毫秒");
    }

    @Async
    public void doTaskThree() throws Exception {
        log.info("开始做任务三");
        long start = System.currentTimeMillis();
        Thread.sleep(random.nextInt(10000));
        long end = System.currentTimeMillis();
        log.info("完成任务三,耗时:" + (end - start) + "毫秒");
    }
}

为了让@Async注解能够生效,还需要在Spring Boot的主程序中配置@EnableAsync

@EnableAsync
@SpringBootApplication
public class Test01Application {
    public static void main(String[] args) {
        SpringApplication.run(Test01Application.class, args);
    }
}

现在测试,可能会遇到不同结果

  • 没有任何任务相关的输出
  • 有部分任务相关的输出
  • 乱序的任务相关的输出

原因是目前doTaskOne、doTaskTwo、doTaskThree三个函数的时候已经是异步执行了。主程序在异步调用之后,主程序并不会理会这三个函数是否执行完成了,由于没有其他需要执行的内容,所以程序就自动结束了,导致了不完整或是没有输出任务相关内容的情况。

注:@Async所修饰的函数不要定义为static类型,这样异步调用不会生效

异步回调方式1

为了让doTaskOne、doTaskTwo、doTaskThree能正常结束,假设我们需要统计一下三个任务并发执行共耗时多少,这就需要等到上述三个函数都完成调动之后记录时间,并计算结果。

那么我们如何判断上述三个异步调用是否已经执行完成呢?我们需要使用CompletableFuture来返回异步调用的结果,就像如下方式改造doTaskOne函数:

@Async
public CompletableFuture<String> doTaskOne() throws Exception {
    log.info("开始做任务一");
    long start = System.currentTimeMillis();
    Thread.sleep(random.nextInt(10000));
    long end = System.currentTimeMillis();
    log.info("完成任务一,耗时:" + (end - start) + "毫秒");
    return CompletableFuture.completedFuture("任务一完成");
}

按照如上方式改造一下其他两个异步函数之后,下面我们改造一下测试用例,让测试在等待完成三个异步调用之后来做一些其他事情。

@Test
public void test() throws Exception {
    long start = System.currentTimeMillis();

    CompletableFuture<String> task1 = asyncTasks.doTaskOne();
    CompletableFuture<String> task2 = asyncTasks.doTaskTwo();
    CompletableFuture<String> task3 = asyncTasks.doTaskThree();

    CompletableFuture.allOf(task1, task2, task3).join();

    long end = System.currentTimeMillis();

    log.info("任务全部完成,总耗时:" + (end - start) + "毫秒");
}

看看结果哪些改变:

  • 在测试用例一开始记录开始时间
  • 在调用三个异步函数的时候,返回CompletableFuture类型的结果对象
  • 通过CompletableFuture.allOf(task1, task2, task3).join()手动实现三个异步任务都结束之前的阻塞效果
  • 三个任务都完成之后,根据结束时间 - 开始时间,计算出三个任务并发执行的总耗时。

可以看到,通过异步调用,让任务一、二、三并发执行,有效的减少了程序的总运行时间。

异步回调方式2

改造doTaskOne方法,其他方法同理

@Async
public CompletableFuture<Long> doTaskOne() throws Exception {
    log.info("开始做任务一");
    long start = System.currentTimeMillis();
    Thread.sleep(random.nextInt(10000));
    long end = System.currentTimeMillis();
    log.info("完成任务一,耗时:" + (end - start) + "毫秒");
    return CompletableFuture.completedFuture(end - start);
}

修改测试方法

@Test
public void test1() throws Exception {
   long start = System.currentTimeMillis();

   CompletableFuture<Long> task1 = asyncTasks.doTaskOne();
   CompletableFuture<Long> task2 = asyncTasks.doTaskTwo();
   CompletableFuture<Long> task3 = asyncTasks.doTaskThree();
   // CompletableFuture.allOf(task1, task2, task3).join();
   Long i = task1.get()+task2.get()+task3.get();
   long end = System.currentTimeMillis();
   log.info("任务全部完成,总耗时:" + (end - start) + "毫秒");
   log.info("任务累计耗时:" + i + "毫秒");
}

和异步回调方式1比较:

  • 没有使用CompletableFuture.allOf(task1, task2, task3).join()主动阻塞,开发中看程序情况决定是否需要使用主动阻塞
  • 使用task1.get(),在获取任务返回结果时会自动阻塞,等待异步线程执行完毕后获取结果
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
@Async和@PostConstruct是Spring框架的两个注解,分别用于实现异步方法和在Bean初始化后执行某些逻辑。 @Async注解用于标记一个方法为异步方法,当该方法被调用时,Spring会将其放入一个线程池进行异步执行,而不会阻塞主线程。这样可以提高系统的并发性能和响应速度。 @PostConstruct注解用于标记一个方法为Bean的初始化方法,在Bean实例化后,Spring会调用被@PostConstruct标记的方法进行一些初始化操作。这个注解通常用于需要在Bean创建之后执行一些逻辑的场景。 在Spring Boot,你可以同时使用@Async和@PostConstruct注解。例如,你可以在一个Bean的初始化方法上使用@PostConstruct注解,在该方法调用一个异步方法,以便在系统启动时执行某些耗时操作。 举个例子,假设你有一个名为UserService的Bean,在它的初始化方法调用一个异步方法来加载用户数据: ```java @Service public class UserService { @Async @PostConstruct public void init() { // 异步加载用户数据 loadData(); } public void loadData() { // 加载用户数据的逻辑 // ... } } ``` 上述代码,当UserService的Bean被实例化后,Spring会调用init方法,在init方法调用loadData方法,由于loadData方法被@Async注解标记,它将异步执行,不会阻塞主线程。 注意,为了让@Async注解生效,你需要在Spring Boot的配置类添加@EnableAsync注解,以启用异步支持。 总结:@Async和@PostConstruct是Spring框架的两个注解,分别用于实现异步方法和在Bean初始化后执行某些逻辑。在Spring Boot,你可以同时使用这两个注解来实现异步初始化操作。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值