1、简介
在Java应用中,绝大多数状况下都是经过同步的方式来实现交互处理的;可是在处理与第三方系统交互的时候,容易形成响应迟缓的状况数据库
2、解决办法
办法一:采用多线程实现异步操做多线程
新建一个线程类:app
public class MyThread extends Thread { @Override public void run() { try { Thread.sleep(5000); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println("。。。。。异步方法调用了。。。。。"); } }
在须要异步操做的地方,开启异步线程:异步
@RequestMapping("/index") public String loginpage() { //开启异步操做 Thread thread1=new Thread(new MyThread()); thread1.start(); return "login"; }
方法二:@Async注解
在Spring中,基于@Async标注的方法,称之为异步方法;这些方法将在执行的时候,将会在独立的线程中被执行,调用者无需等待它的完成,便可继续其余的操做。
(1)基于Java配置的启用方式:
@Configuration
@EnableAsync
public class SpringAsyncConfig { ... } async
(2)基于XML配置文件的启用方式
//加载bean
<bean id="asyncTest" class="com.utils.AsyncTest" lazy-init="false"/>
<task:executor id="myexecutor"pool-size="5"/>
<task:annotation-driven executor="myexecutor"/>ide
3、实际使用
(1)新建一个异步操做类测试
@Component public class AsyncTest { @Async("myexecutor") public void sayHello(){ try { Thread.sleep(3000); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println("hello...hello"); } }
(2)在须要异步的地方,使用上述方法:spa
@Autowired private AsyncTest asyncTest; @RequestMapping("/index") public String loginpage() { System.out.println("开始调用 asyncTest.sayHello()"); asyncTest.sayHello(); System.out.println("结束调用 asyncTest.sayHello()"); return "login"; }
上述方法返回值是void的演示,还有存在返回值的异步操做;线程
改写sayHello()方法 @Async("myexecutor") public Future<String> sayHello(){ try { Thread.sleep(3000); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println("hello...hello"); return new AsyncResult<String>("这是sayHello()返回值"); }
打印sayHello()返回值 @RequestMapping("/index") public String loginpage() throws ExecutionException, InterruptedException { System.out.println("开始调用 asyncTest.sayHello()"); Future<String> results=asyncTest.sayHello(); System.out.println("结束调用 asyncTest.sayHello()"); while(true) { //这里使用了循环判断,等待获取结果信息 if (results.isDone()) { //判断是否执行完毕 System.out.println("Result from asynchronous process - " + results.get()); break; } } return "login"; }
获取异步方法的结果信息,是经过不停的检查Future的状态来获取当前的异步方法是否执行完毕来实现的
经实际测试发现,若是打印异步操做返回值,则是等待异步操做执行结束后才能得到返回值,如此一来整个流程下来花费时间和同步方法同样。code
4、@Async调用中的事务处理机制
在@Async标注的方法,同时也适用了@Transactional进行了标注;在其调用数据库操做之时,将没法产生事务管理的控制,缘由就在于其是基于异步处理的操做。那该如何给这些操做添加事务管理呢?能够将须要事务管理操做的方法放置到异步方法内部,在内部被调用的方法上添加@Transactional.
例如:
方法A,使用了@Async/@Transactional来标注,可是没法产生事务控制的目的。 方法B,使用了@Async来标注, B中调用了C、D,C/D分别使用@Transactional作了标注,则可实现事务控制的目的。