zoukankan      html  css  js  c++  java
  • Spring中@Async注解实现“方法”的异步调用

    原文:http://www.cnblogs.com/zhengbin/p/6104502.html

    简单介绍:

    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秒
  • 相关阅读:
    @SneakyThrows
    docker部署elasticsearch
    docker部署rabbitmq
    docker部署minio
    docker 部署 jenkins
    linux 根据文件名全局查找位置
    docker 容器与宿主机之间文件拷贝
    excel 查看当前单元格是否存在某一列
    机器学习sklearn
    一些博客链接
  • 原文地址:https://www.cnblogs.com/shihaiming/p/6237008.html
Copyright © 2011-2022 走看看