详解SpringBoot中异步请求和异步调用(看完这一篇就够了)

发布时间:2020-09-07 14:52:05 作者:会炼钢的小白龙
来源:脚本之家 阅读:242

一、SpringBoot中异步请求的使用

1、异步请求与同步请求

详解SpringBoot中异步请求和异步调用(看完这一篇就够了)

详解SpringBoot中异步请求和异步调用(看完这一篇就够了)

特点:

可以先释放容器分配给请求的线程与相关资源,减轻系统负担,释放了容器所分配线程的请求,其响应将被延后,可以在耗时处理完成(例如长时间的运算)时再对客户端进行响应。一句话:增加了服务器对客户端请求的吞吐量(实际生产上我们用的比较少,如果并发请求量很大的情况下,我们会通过nginx把请求负载到集群服务的各个节点上来分摊请求压力,当然还可以通过消息队列来做请求的缓冲)。

2、异步请求的实现

方式一:Servlet方式实现异步请求

@RequestMapping(value = "/email/servletReq", method = GET)
 public void servletReq (HttpServletRequest request, HttpServletResponse response) {
  AsyncContext asyncContext = request.startAsync();
  //设置监听器:可设置其开始、完成、异常、超时等事件的回调处理
 asyncContext.addListener(new AsyncListener() {
   @Override
   public void onTimeout(AsyncEvent event) throws IOException {
    System.out.println("超时了...");
    //做一些超时后的相关操作...
   }
   @Override
   public void onStartAsync(AsyncEvent event) throws IOException {
    System.out.println("线程开始");
   }
   @Override
   public void onError(AsyncEvent event) throws IOException {
    System.out.println("发生错误:"+event.getThrowable());
   }
   @Override
   public void onComplete(AsyncEvent event) throws IOException {
    System.out.println("执行完成");
    //这里可以做一些清理资源的操作...
   }
  });
  //设置超时时间
  asyncContext.setTimeout(20000);
  asyncContext.start(new Runnable() {
   @Override
   public void run() {
    try {
     Thread.sleep(10000);
     System.out.println("内部线程:" + Thread.currentThread().getName());
     asyncContext.getResponse().setCharacterEncoding("utf-8");
     asyncContext.getResponse().setContentType("text/html;charset=UTF-8");
     asyncContext.getResponse().getWriter().println("这是异步的请求返回");
    } catch (Exception e) {
     System.out.println("异常:"+e);
    }
    //异步请求完成通知
    //此时整个请求才完成
    asyncContext.complete();
   }
  });
  //此时之类 request的线程连接已经释放了
  System.out.println("主线程:" + Thread.currentThread().getName());
 }

方式二:使用很简单,直接返回的参数包裹一层callable即可,可以继承WebMvcConfigurerAdapter类来设置默认线程池和超时处理

@RequestMapping(value = "/email/callableReq", method = GET)
 @ResponseBody
 public Callable<String> callableReq () {
  System.out.println("外部线程:" + Thread.currentThread().getName());
  return new Callable<String>() {
   @Override
   public String call() throws Exception {
    Thread.sleep(10000);
    System.out.println("内部线程:" + Thread.currentThread().getName());
    return "callable!";
   }
  };
 }
 @Configuration
 public class RequestAsyncPoolConfig extends WebMvcConfigurerAdapter {
 @Resource
 private ThreadPoolTaskExecutor myThreadPoolTaskExecutor;
 @Override
 public void configureAsyncSupport(final AsyncSupportConfigurer configurer) {
  //处理 callable超时
  configurer.setDefaultTimeout(60*1000);
  configurer.setTaskExecutor(myThreadPoolTaskExecutor);
  configurer.registerCallableInterceptors(timeoutCallableProcessingInterceptor());
 }
 @Bean
 public TimeoutCallableProcessingInterceptor timeoutCallableProcessingInterceptor() {
  return new TimeoutCallableProcessingInterceptor();
 }
}

方式三:和方式二差不多,在Callable外包一层,给WebAsyncTask设置一个超时回调,即可实现超时处理

@RequestMapping(value = "/email/webAsyncReq", method = GET)
@ResponseBody
public WebAsyncTask<String> webAsyncReq () {
 System.out.println("外部线程:" + Thread.currentThread().getName());
 Callable<String> result = () -> {
  System.out.println("内部线程开始:" + Thread.currentThread().getName());
  try {
   TimeUnit.SECONDS.sleep(4);
  } catch (Exception e) {
   // TODO: handle exception
  }
  logger.info("副线程返回");
  System.out.println("内部线程返回:" + Thread.currentThread().getName());
  return "success";
 };
 WebAsyncTask<String> wat = new WebAsyncTask<String>(3000L, result);
 wat.onTimeout(new Callable<String>() {

  @Override
  public String call() throws Exception {
   // TODO Auto-generated method stub
   return "超时";
  }
 });
 return wat;
}

方式四:DeferredResult可以处理一些相对复杂一些的业务逻辑,最主要还是可以在另一个线程里面进行业务处理及返回,即可在两个完全不相干的线程间的通信。

@RequestMapping(value = "/email/deferredResultReq", method = GET)
@ResponseBody
public DeferredResult<String> deferredResultReq () {
 System.out.println("外部线程:" + Thread.currentThread().getName());
 //设置超时时间
 DeferredResult<String> result = new DeferredResult<String>(60*1000L);
 //处理超时事件 采用委托机制
 result.onTimeout(new Runnable() {

  @Override
  public void run() {
   System.out.println("DeferredResult超时");
   result.setResult("超时了!");
  }
 });
 result.onCompletion(new Runnable() {

  @Override
  public void run() {
   //完成后
   System.out.println("调用完成");
  }
 });
 myThreadPoolTaskExecutor.execute(new Runnable() {

  @Override
  public void run() {
   //处理业务逻辑
   System.out.println("内部线程:" + Thread.currentThread().getName());
   //返回结果
   result.setResult("DeferredResult!!");
  }
 });
 return result;
}

二、SpringBoot中异步调用的使用

1、介绍

异步请求的处理。除了异步请求,一般上我们用的比较多的应该是异步调用。通常在开发过程中,会遇到一个方法是和实际业务无关的,没有紧密性的。比如记录日志信息等业务。这个时候正常就是启一个新线程去做一些业务处理,让主线程异步的执行其他业务。

2、使用方式(基于spring下)

3、注意事项

4、什么情况下会导致@Async异步方法会失效?

调用同一个类下注有@Async异步方法:在spring中像@Async和@Transactional、cache等注解本质使用的是动态代理,其实Spring容器在初始化的时候Spring容器会将含有AOP注解的类对象“替换”为代理对象(简单这么理解),那么注解失效的原因就很明显了,就是因为调用方法的是对象本身而不是代理对象,因为没有经过Spring容器,那么解决方法也会沿着这个思路来解决。

5、解决4中问题1的方式(其它2,3两个问题自己注意下就可以了)

将要异步执行的方法单独抽取成一个类,原理就是当你把执行异步的方法单独抽取成一个类的时候,这个类肯定是被Spring管理的,其他Spring组件需要调用的时候肯定会注入进去,这时候实际上注入进去的就是代理类了。

其实我们的注入对象都是从Spring容器中给当前Spring组件进行成员变量的赋值,由于某些类使用了AOP注解,那么实际上在Spring容器中实际存在的是它的代理对象。那么我们就可以通过上下文获取自己的代理对象调用异步方法。

@Controller
@RequestMapping("/app")
public class EmailController {
 //获取ApplicationContext对象方式有多种,这种最简单,其它的大家自行了解一下
 @Autowired
 private ApplicationContext applicationContext;
 
  @RequestMapping(value = "/email/asyncCall", method = GET)
  @ResponseBody
  public Map<String, Object> asyncCall () {
    Map<String, Object> resMap = new HashMap<String, Object>();
    try{
  //这样调用同类下的异步方法是不起作用的
      //this.testAsyncTask();
  //通过上下文获取自己的代理对象调用异步方法
   EmailController emailController = (EmailController)applicationContext.getBean(EmailController.class);
   emailController.testAsyncTask();
      resMap.put("code",200);
    }catch (Exception e) {
  resMap.put("code",400);
      logger.error("error!",e);
    }
    return resMap;
  }
 //注意一定是public,且是非static方法
 @Async
  public void testAsyncTask() throws InterruptedException {
    Thread.sleep(10000);
    System.out.println("异步任务执行完成!");
  }
}

开启cglib代理,手动获取Spring代理类,从而调用同类下的异步方法。

首先,在启动类上加上@EnableAspectJAutoProxy(exposeProxy = true)注解。

代码实现,如下:

@Service
@Transactional(value = "transactionManager", readOnly = false, propagation = Propagation.REQUIRED, rollbackFor = Throwable.class)
public class EmailService {
  @Autowired
  private ApplicationContext applicationContext;
  @Async
  public void testSyncTask() throws InterruptedException {
    Thread.sleep(10000);
    System.out.println("异步任务执行完成!");
  }
  public void asyncCallTwo() throws InterruptedException {
    //this.testSyncTask();
//    EmailService emailService = (EmailService)applicationContext.getBean(EmailService.class);
//    emailService.testSyncTask();
    boolean isAop = AopUtils.isAopProxy(EmailController.class);//是否是代理对象;
    boolean isCglib = AopUtils.isCglibProxy(EmailController.class); //是否是CGLIB方式的代理对象;
    boolean isJdk = AopUtils.isJdkDynamicProxy(EmailController.class); //是否是JDK动态代理方式的代理对象;
    //以下才是重点!!!
 EmailService emailService = (EmailService)applicationContext.getBean(EmailService.class);
    EmailService proxy = (EmailService) AopContext.currentProxy();
    System.out.println(emailService == proxy ? true : false);
    proxy.testSyncTask();
    System.out.println("end!!!");
  }
}

三、异步请求与异步调用的区别

四、总结

总结

以上所述是小编给大家介绍的SpringBoot中异步请求和异步调用问题,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对亿速云网站的支持!

推荐阅读:
  1. Python 中的 or and 运算,看这一篇就够
  2. 关于jquery同步和异步请求问题总结

免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。

spring boot 异步

上一篇:Spring Boot 项目创建的详细步骤(图文)

下一篇:python 队列基本定义与使用方法【初始化、赋值、判断等】

相关阅读

您好,登录后才能下订单哦!

密码登录
登录注册
其他方式登录
点击 登录注册 即表示同意《亿速云用户服务条款》