这里讲解一下Spring对异步事件机制的支持,实现方式有两种:git
一、全局异步
即只要是触发事件都是以异步执行,具体配置(spring-config-register.xml)以下:github
Java代码
- <task:executor id="executor" pool-size="10" />
- <!-- 名字必须是applicationEventMulticaster和messageSource是同样的,默认找这个名字的对象 -->
- <!-- 名字必须是applicationEventMulticaster,由于AbstractApplicationContext默认找个 -->
- <!-- 若是找不到就new一个,但不是异步调用而是同步调用 -->
- <bean id="applicationEventMulticaster" class="org.springframework.context.event.SimpleApplicationEventMulticaster">
- <!-- 注入任务执行器 这样就实现了异步调用(缺点是全局的,要么所有异步,要么所有同步(删除这个属性便是同步)) -->
- <property name="taskExecutor" ref="executor"/>
- </bean>
经过注入taskExecutor来完成异步调用。具体实现可参考以前的代码介绍。这种方式的缺点很明显:要么你们都是异步,要么你们都不是。因此不推荐使用这种方式。
二、更灵活的异步支持
spring3提供了@Aync注解来完成异步调用。此时咱们可使用这个新特性来完成异步调用。不只支持异步调用,还支持简单的任务调度,好比个人项目就去掉Quartz依赖,直接使用spring3这个新特性,具体可参考spring-config.xml。web
2.一、开启异步调用支持spring
Java代码
- <!-- 开启@AspectJ AOP代理 -->
- <aop:aspectj-autoproxy proxy-target-class="true"/>
-
- <!-- 任务调度器 -->
- <task:scheduler id="scheduler" pool-size="10"/>
-
- <!-- 任务执行器 -->
- <task:executor id="executor" pool-size="10"/>
-
- <!--开启注解调度支持 @Async @Scheduled-->
- <task:annotation-driven executor="executor" scheduler="scheduler" proxy-target-class="true"/>
2.二、配置监听器让其支持异步调用spring-mvc
Java代码
- @Component
- public class EmailRegisterListener implements ApplicationListener<RegisterEvent> {
- @Async
- @Override
- public void onApplicationEvent(final RegisterEvent event) {
- System.out.println("注册成功,发送确认邮件给:" + ((User)event.getSource()).getUsername());
- }
- }
使用@Async注解便可,很是简单。 mvc
这样不只能够支持经过调用,也支持异步调用,很是的灵活,实际应用推荐你们使用这种方式。app
经过如上,大致了解了Spring的事件机制,可使用该机制很是简单的完成如注册流程,并且对于比较耗时的调用,能够直接使用Spring自身的异步支持来优化。异步
1 <beans xmlns="http://www.springframework.org/schema/beans"
2 xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
3 xmlns:tx="http://www.springframework.org/schema/tx"
4 xmlns:context="http://www.springframework.org/schema/context"
5 xmlns:aop="http://www.springframework.org/schema/aop"
6 xmlns:mvc="http://www.springframework.org/schema/mvc"
7 xmlns:task="http://www.springframework.org/schema/task"
8 xsi:schemaLocation="http://www.springframework.org/schema/beans
9 http://www.springframework.org/schema/beans/spring-beans-4.0.xsd
10 http://www.springframework.org/schema/tx
11 http://www.springframework.org/schema/tx/spring-tx-4.0.xsd
12 http://www.springframework.org/schema/context
13 http://www.springframework.org/schema/context/spring-context-4.0.xsd
14 http://www.springframework.org/schema/mvc
15 http://www.springframework.org/schema/mvc/spring-mvc-4.0.xsd
16 http://www.springframework.org/schema/task
17 http://www.springframework.org/schema/task/spring-task-4.0.xsd">
18
19 <context:annotation-config />
20 <!--扫描注解 -->
21 <context:component-scan base-package="com.tf" />
22 <!-- 支持异步方法执行 -->
23 <task:annotation-driven />
View Code
这个注解用于标注某个方法或某个类里面的全部方法都是须要异步处理的。被注解的方法被调用的时候,会在新线程中执行,而调用它的方法会在原来的线程中执行。这样能够避免阻塞、以及保证任务的实时性。适用于处理log、发送邮件、短信……等。
注解的应用范围:
- 类:表示这个类中的全部方法都是异步的
- 方法:表示这个方法是异步的,若是类也注解了,则以这个方法的注解为准
相关的配置:
<task:annotation-driven />配置:
- executor:指定一个缺省的executor给@Async使用。
例子:
<task:annotation-driven executor="asyncExecutor" />
<task:executor />配置参数:
- id:当配置多个executor时,被@Async("id")指定使用;也被做为线程名的前缀。
- pool-size:
- core size:最小的线程数,缺省:1
- max size:最大的线程数,缺省:Integer.MAX_VALUE
- queue-capacity:当最小的线程数已经被占用满后,新的任务会被放进queue里面,当这个 queue的capacity也被占满以后,pool里面会建立新线程处理这个任务,直到总线程数达到了max size,这时系统会拒绝这个任务并抛出TaskRejectedException异常(缺省配置的状况下,能够经过rejection-policy 来决定如何处理这种状况)。缺省值为:Integer.MAX_VALUE
- keep-alive:超过core size的那些线程,任务完成后,再通过这个时长(秒)会被结束掉
- rejection-policy:当pool已经达到max size的时候,如何处理新任务
- ABORT(缺省):抛出TaskRejectedException异常,而后不执行
- DISCARD:不执行,也不抛出异常
- DISCARD_OLDEST:丢弃queue中最旧的那个任务
- CALLER_RUNS:不在新线程中执行任务,而是有调用者所在的线程来执行