spring boot异步(Async)任务调度实现方法

在没有使用spring boot之前,我们的做法是在配置文件中定义一个任务池,然后将@Async注解的任务丢到任务池中去执行,那么在spring boot中,怎么来实现异步任务的调用了,方法更简单。

我们还是结合前面

spring boot整合JMS(ActiveMQ实现)

这篇博客里面的代码来实现。

一、功能说明

消费者在监听到队列里面的消息时,将接收消息的任务作为异步任务处理。

二、代码修改

消费者1:

package com.chhliu.springboot.jms;
import org.springframework.jms.annotation.JmsListener;
import org.springframework.scheduling.annotation.Async;
import org.springframework.stereotype.Component;
@Component
public class Consumer {
 @JmsListener(destination = "mytest.queue")
 @Async //该方法会异步执行,也就是说主线程会直接跳过该方法,而是使用线程池中的线程来执行该方法
 public void receiveQueue(String text) {
  System.out.println(Thread.currentThread().getName()+":Consumer收到的报文为:"+text);
 }
} 

消费者2:

package com.chhliu.springboot.jms;
import org.springframework.jms.annotation.JmsListener;
import org.springframework.messaging.handler.annotation.SendTo;
import org.springframework.stereotype.Component;
@Component
public class Consumer2 {
 @JmsListener(destination = "mytest.queue")
 @SendTo("out.queue")
 public String receiveQueue(String text) {
  System.out.println(Thread.currentThread().getName()+":Consumer2收到的报文为:"+text);
  return "return message"+text;
 }
} 

在测试类上添加如下注解:

package com.chhliu.springboot.jms;
import javax.jms.Destination;
import org.apache.activemq.command.ActiveMQQueue;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.scheduling.annotation.EnableAsync;
import org.springframework.test.context.junit4.SpringRunner;
@RunWith(SpringRunner.class)
@SpringBootTest
@EnableAsync // 开启异步任务支持
public class SpringbootJmsApplicationTests {
 @Autowired
 private Producer producer;
 @Test
 public void contextLoads() throws InterruptedException {
  Destination destination = new ActiveMQQueue("mytest.queue");
  for(int i=0; i<100; i++){
   producer.sendMessage(destination, "myname is chhliu!!!");
  }
 }
} 

三、测试结果

DefaultMessageListenerContainer-1:Consumer2收到的报文为:myname is chhliu!!!
从out.queue队列收到的回复报文为:return messagemyname is chhliu!!!
SimpleAsyncTaskExecutor-45:Consumer收到的报文为:myname is chhliu!!!
DefaultMessageListenerContainer-1:Consumer2收到的报文为:myname is chhliu!!!
从out.queue队列收到的回复报文为:return messagemyname is chhliu!!!
SimpleAsyncTaskExecutor-46:Consumer收到的报文为:myname is chhliu!!!
DefaultMessageListenerContainer-1:Consumer2收到的报文为:myname is chhliu!!!
从out.queue队列收到的回复报文为:return messagemyname is chhliu!!!
SimpleAsyncTaskExecutor-47:Consumer收到的报文为:myname is chhliu!!!
DefaultMessageListenerContainer-1:Consumer2收到的报文为:myname is chhliu!!!
从out.queue队列收到的回复报文为:return messagemyname is chhliu!!!
SimpleAsyncTaskExecutor-48:Consumer收到的报文为:myname is chhliu!!!
DefaultMessageListenerContainer-1:Consumer2收到的报文为:myname is chhliu!!!
从out.queue队列收到的回复报文为:return messagemyname is chhliu!!!
SimpleAsyncTaskExecutor-49:Consumer收到的报文为:myname is chhliu!!!
DefaultMessageListenerContainer-1:Consumer2收到的报文为:myname is chhliu!!!
从out.queue队列收到的回复报文为:return messagemyname is chhliu!!!
SimpleAsyncTaskExecutor-50:Consumer收到的报文为:myname is chhliu!!!
DefaultMessageListenerContainer-1:Consumer2收到的报文为:myname is chhliu!!! 

从上面的测试结果可以看出,由于消费者2没有使用异步任务方式,所以消费者2消费消息都是由固定的线程DefaultMessageListenerContainer-1这个线程来处理的,而消费者1由于使用了异步任务的方式,每次处理接收到的消息都是由不同的线程来处理的,当接收到消息时,直接将任务丢到任务池中去处理,而主线程则继续跑,从测试结果中还可以推断出,spring boot默认使用了newCachedThreadPool线程池来实现。

关于线程池的具体用法,请参考我的另一篇博文:http://www.jb51.net/article/134870.htm

四、异步任务有返回

在实际的开发中,我们会经常遇到异步任务有返回的情况,那么在spring boot中,怎么来实现了?

下面以异步发邮件为例,来进行说明,示例代码如下:

@Async("taskExecutePool") // 异步任务会提交到taskExecutePool任务池中执行
 public Future<Response> doSendEmail(MailInfo mailInfo) {// 异步任务返回,使用Future<Response>来异步返回
  log.info(Thread.currentThread().getName()+"调用了doSendEmail异步方法!");
  SendMailSession session = null;
  Response res = new Response();
  boolean isOK = sendEmail(mailInfo);// 具体发邮件的方法
 if(isOK){
  res.setSuccess(true);
 }else{
  res.setSuccess(false);
 }
 return new AsyncResult<Response>(res); 

返回之后怎么使用?示例代码如下:

Future<Response> result = taskJob.doSendEmail(mailInfo);
   res = result.get(6, TimeUnit.SECONDS); 

这样就可以获取到异步任务的返回了!

总结

以上所述是小编给大家介绍的spring boot异步(Async)任务调度实现方法,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对我们网站的支持!

您可能感兴趣的文章:

  • Java任务调度的常见实现方法与比较详解
  • Java中Spring使用Quartz任务调度定时器
  • java使用任务架构执行任务调度示例
  • java多线程并发executorservice(任务调度)类
  • Spring整合Quartz实现定时任务调度的方法
  • Spring整合TimerTask实现定时任务调度
  • Spring内置任务调度如何实现添加、取消与重置详解
  • springboot+Quartz实现任务调度的示例代码
  • Spring 中使用Quartz实现任务调度
  • 使用java.util.Timer实现任务调度
(0)

相关推荐

  • java使用任务架构执行任务调度示例

    复制代码 代码如下: package com.yao; import java.util.concurrent.Callable;import java.util.concurrent.ExecutionException;import java.util.concurrent.ExecutorService;import java.util.concurrent.Executors;import java.util.concurrent.Future;import java.util.conc

  • Java任务调度的常见实现方法与比较详解

    本文实例讲述了Java任务调度的常见实现方法与比较.分享给大家供大家参考,具体如下: 简介: 综观目前的 Web 应用,多数应用都具备任务调度的功能.本文由浅入深介绍了几种任务调度的 Java 实现方法,包括 Timer,Scheduler, Quartz 以及 JCron Tab,并对其优缺点进行比较,目的在于给需要开发任务调度的程序员提供有价值的参考. 任务调度是指基于给定时间点,给定时间间隔或者给定执行次数自动执行任务.这里由浅入深介绍四种任务调度的 Java 实现: Timer Sche

  • Spring内置任务调度如何实现添加、取消与重置详解

    前言 大家应该都有所体会,使用Spring的任务调度给我们的开发带来了极大的便利,不过当我们的任务调度配置完成后,很难再对其进行更改,除非停止服务器,修改配置,然后再重启,显然这样是不利于线上操作的,为了实现动态的任务调度修改,我在网上也查阅了一些资料,大部分都是基于quartz实现的,使用Spring内置的任务调度则少之又少,而且效果不理想,需要在下次任务执行后,新的配置才能生效,做不到立即生效.本着探索研究的原则,查看了一下Spring的源码,下面为大家提供一种Spring内置任务调度实现添

  • Spring整合Quartz实现定时任务调度的方法

    最近项目中需要实现定时执行任务,比如定时计算会员的积分.调用第三方接口等,由于项目采用spring框架,所以这里结合spring框架来介绍. 编写作业类 即普通的pojo,如下: package com.pcmall.task; import org.slf4j.Logger; import org.slf4j.LoggerFactory; public class TaskA { private static Logger logger = LoggerFactory.getLogger(Ta

  • springboot+Quartz实现任务调度的示例代码

    在spring框架中通过 @Schedule 可以实现定时任务,通过该注解 cron 的属性描述的规则,spring会去调用这个方法. spring已经简单粗暴的实现了定时任务,为什么要使用Quartz ? 如果你现在有很多个定时任务,规则不同,例如: 想每月25号,信用卡自动还款 想每年4月1日自己给当年暗恋女神发一封匿名贺卡 想每隔1小时,备份一下自己的爱情动作片 学习笔记到云盘 maven 依赖 <dependency> <groupId>org.quartz-schedul

  • 使用java.util.Timer实现任务调度

    任务调度是指基于给定时间点,给定时间间隔或者给定执行次数自动执行任务. 举个例子,比如说我们希望一个系统每周日晚上9点都将数据库文件备份一次,这时我们就可以使用任务调度来实现.为了更加的方便,我们需要在tomcat启动后,自动开始这个调度. 下面是TimerTask的API: 下面是Timer类的API 下面的例子中实现了两个功能: 1.监测tomcat的web容器的启动与关闭 2.当web容器启动后,任务调度分配任务对象,时间和周期. 为了监测web容器的变化,首先需要在web.xml中注册监

  • Java中Spring使用Quartz任务调度定时器

    Quartz 任务调度是什么 Quartz 是 OpenSymphony 开源组织在 Job scheduling 领域又一个开源项目,它可以与 J2EE 与 J2SE 应用程序相结合也可以单独使用.Quartz 是一个完全由 Java 编写的开源作业调度框架.不要让作业调度这个术语吓着你.尽管Quartz框架整合了许多额外功能,但就其简易形式看,你会发现它易用得简直让人受不了! 其实,他还是没有解释明白,我简单说一下:Quartz 作业调度就是可以实现定时任务.它可以实现类似 Windows

  • java多线程并发executorservice(任务调度)类

    复制代码 代码如下: package com.yao; import java.util.concurrent.Executors;import java.util.concurrent.ScheduledExecutorService;import java.util.concurrent.ScheduledFuture;import java.util.concurrent.TimeUnit; /** * 以下是一个带方法的类,它设置了 ScheduledExecutorService ,2

  • Spring整合TimerTask实现定时任务调度

    一. 前言 最近在公司的项目中用到了定时任务, 本篇博文将会对TimerTask定时任务进行总结, 其实TimerTask在实际项目中用的不多, 因为它不能在指定时间运行, 只能让程序按照某一个频度运行. 二. TimerTask JDK中Timer是一个定时器类, 它可以为指定的定时任务进行配置. JDK中TimerTask是一个定时任务类, 该类实现了Runnable接口, 是一个抽象类, 我们可以继承这个类, 实现定时任务. /** * 继承TimerTask实现定时任务 */ publi

  • Spring 中使用Quartz实现任务调度

    前言:Spring中使用Quartz 有两种方式,一种是继承特定的基类:org.springframework.scheduling.quartz.QuartzJobBean,另一种则不需要,(推荐使用第二种).下面分别介绍. 1.作业类继承 org.springframework.scheduling.quartz.QuartzJobBean 第一步:定义作业类 java代码 import java.text.SimpleDateFormat; import java.util.Date; i

随机推荐