欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

Spring @async方法如何添加注解实现异步调用

程序员文章站 2022-10-28 13:18:17
这篇文章主要介绍了spring @async方法如何添加注解实现异步调用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 在我们使用...

这篇文章主要介绍了spring @async方法如何添加注解实现异步调用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下

在我们使用spring框架的过程中,在很多时候我们会使用@async注解来异步执行某一些方法,提高系统的执行效率。今天我们来探讨下 spring 是如何完成这个功能的。

spring 在扫描bean的时候会扫描方法上是否包含@async的注解,如果包含的,spring会为这个bean动态的生成一个子类,我们称之为代理类(?), 代理类是继承我们所写的bean的,然后把代理类注入进来,那此时,在执行此方法的时候,会到代理类中,代理类判断了此方法需要异步执行,就不会调用父类 (我们原本写的bean)的对应方法。spring自己维护了一个队列,他会把需要执行的方法,放入队列中,等待线程池去读取这个队列,完成方法的执行, 从而完成了异步的功能。

我们可以关注到再配置task的时候,是有参数让我们配置线程池的数量的。因为这种实现方法,所以在同一个类中的方法调用,添加@async注解是失效的!,原因是当你在同一个类中的时候,方法调用是在类体内执行的,spring无法截获这个方法调用。

那在深入一步,spring为我们提供了aop,面向切面的功能。他的原理和异步注解的原理是类似的,spring在启动容器的时候,会扫描切面所定义的 类。在这些类被注入的时候,所注入的也是代理类,当你调用这些方法的时候,本质上是调用的代理类。通过代理类再去执行父类相对应的方法,那spring只 需要在调用之前和之后执行某段代码就完成了aop的实现了!

那最后我们还有一个问题,spring是如何动态的生成某一个类的子类的?代理类?

简单介绍:

spring为任务调度与异步方法执行提供了注解支持。通过在方法上设置@async注解,可使得方法被异步调用。也就是说调用者会在调用时立即返回,而被调用方法的实际执行是交给spring的taskexecutor来完成。

开启@async注解:

<task:annotation-driven executor="annotationexecutor" />
<!-- 支持 @async 注解 -->
<task:executor id="annotationexecutor" pool-size="20"/>

同时加入<context:component-scan />扫描注解。

为了比较,先来一个同步调用:

@component
public class testasyncbean {
  public void sayhello4() throws interruptedexception {
    thread.sleep(2 * 1000);//网络连接中 。。。消息发送中。。。
    system.out.println("我爱你啊!");
}
@runwith(springjunit4classrunner.class)
@contextconfiguration({"classpath:/applicationcontext.xml"})
public class testasync {
  @test
  public void test_sayhello4() throws interruptedexception, executionexception {
    system.out.println("你不爱我了么?");
    testasyncbean.sayhello4();
    system.out.println("回的这么慢, 你肯定不爱我了, 我们还是分手吧。。。");
    thread.sleep(3 * 1000);// 不让主进程过早结束
  }
}

输出结果:

你不爱我了么?
我爱你啊!
回的这么慢, 你肯定不爱我了, 我们还是分手吧。。。 

 同步调用会按代码顺序依次进行下去,如果哪里需要等待,那么就阻塞在那里,不再向下继续进行。

使用@async的异步调用:

@component
public class testasyncbean {
  @async
  public void sayhello3() throws interruptedexception {
    thread.sleep(2 * 1000);//网络连接中 。。。消息发送中。。。
    system.out.println("我爱你啊!");
  }
}
@runwith(springjunit4classrunner.class)
@contextconfiguration({"classpath:/applicationcontext.xml"})
public class testasync {
  @autowired
  private testasyncbean testasyncbean;
  @test
  public void test_sayhello3() throws interruptedexception, executionexception {
    system.out.println("你不爱我了么?");
    testasyncbean.sayhello3();
    system.out.println("你竟无话可说, 我们分手吧。。。");
    thread.sleep(3 * 1000);// 不让主进程过早结束
  }
}

输出结果:

你不爱我了么?
你竟无话可说, 我们分手吧。。。
我爱你啊!

异步调用,通过开启新的线程来执行调用的方法,不影响主线程。异步方法实际的执行交给了spring的taskexecutor来完成。

上面这种方式是没有返回值的,下面尝试有返回值的异步调用:

@component
public class testasyncbean {
  @async
  public string sayhello2() throws interruptedexception {
    thread.sleep(2 * 1000);//网络连接中 。。。消息发送中。。。
    return "我爱你啊!";// 调用方调用后会立即返回,所以返回null
  }
}
@runwith(springjunit4classrunner.class)
@contextconfiguration({"classpath:/applicationcontext.xml"})
public class testasync {
  @autowired
  private testasyncbean testasyncbean;
  @test
  public void test_sayhello2() throws interruptedexception, executionexception {
    system.out.println("你不爱我了么?");
    system.out.println(testasyncbean.sayhello2());
    system.out.println("你说的啥? 我们还是分手吧。。。");
    thread.sleep(3 * 1000);// 不让主进程过早结束
  }
}

输出结果:

你不爱我了么?
null
你说的啥? 我们还是分手吧。。。

通过直接获取返回值得方式是不行的,这里就需要用到异步回调,异步方法返回值必须为future<>,就像callable与future。

下面通过asyncresult<>来获得异步调用的返回值:

@component
public class testasyncbean {
  @async
  public future<string> sayhello1() throws interruptedexception {
    int thinking = 2;
    thread.sleep(thinking * 1000);//网络连接中 。。。消息发送中。。。
    system.out.println("我爱你啊!");
    return new asyncresult<string>("发送消息用了"+thinking+"秒");
  }
}
@runwith(springjunit4classrunner.class)
@contextconfiguration({"classpath:/applicationcontext.xml"})
public class testasync {
  @autowired
  private testasyncbean testasyncbean;
  @test
  public void test_sayhello1() throws interruptedexception, executionexception {
    future<string> future = null;
    system.out.println("你不爱我了么?");
    future = testasyncbean.sayhello1();
    system.out.println("你竟无话可说, 我们分手吧。。。");
    thread.sleep(3 * 1000);// 不让主进程过早结束
    system.out.println(future.get());
  }
}

输出结果:

你不爱我了么?
你竟无话可说, 我们分手吧。。。
我爱你啊!

发送消息用了2秒

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。