ApplicationEvent&ApplicationListener使用
在SpringBoot使用ApplicationEvent&ApplicationListener完成业务解耦
前言:
项目中往往各个业务逻辑之间耦合性较强,因为我们在service都是直接引用的关联service或者jpa来作为协作处理逻辑,然而这种方式在后期更新、维护性难度都是大大提高了。通过使用事件通知、事件监听形式来处理逻辑时耦合性则是可以降到最小。
spring中事件监听实现
ApplicationEvent、ApplicationListener是Spring为我们提供的一个事件监听、订阅的实现,内部实现原理是观察者设计模式,设计初衷也是为了系统业务逻辑之间的解耦,提高可扩展性以及可维护性。事件发布者并不需要考虑谁去监听,监听具体的实现内容是什么,发布者的工作只是为了发布事件而已。
下面我们通过使用ApplicationEvent和ApplicationListener来完成简单的用户注册,以及注册成功后发送邮件的例子。
1、开始我们的示例之前,我们先定义一个User对象
public class User implements java.io.Serializable{
private static final long serialVersionUID = 110406535439289650L;
private String username;
private String password;
//省略get/set方法
}
2、创建UserRegisterEvent
定义一个用户注册事件,监听都是围绕着事件来挂起的。
public class UserRegisterEvent extends ApplicationEvent{
private static final long serialVersionUID = -2493647928101170759L;
//注册用户事件
private User user;
//重写构造函数
public UserRegisterEvent(Object source,User user) {
super(source);
this.user = user;
}
public User getUser() {
return user;
}
public void setUser(User user) {
this.user = user;
}
}
我们自定义事件UserRegisterEvent继承了ApplicationEvent,继承后必须重载构造函数,构造函数的参数可以任意指定,其中source参数指的是发生事件的对象,一般我们在发布事件时使用的是this关键字代替本类对象,而user参数是我们自定义的注册用户对象,该对象可以在监听内被获取。
3、创建UserService
UserService类中添加一个registerUser方法,该方法实现注册事件发布功能
@Service
public class UserService {
@Autowired
private ApplicationContext applicationContext;
public void registerUser(User user){
//发布注册事件
applicationContext.publishEvent(new UserRegisterEvent(this, user));
}
}
事件发布是由ApplicationContext对象管控的,我们发布事件前需要注入ApplicationContext对象调用publishEvent方法完成事件发布。
4、创建UserController
创建一个@RestController控制器,对应添加一个注册方法简单实现
@RestController
public class UserController {
@Autowired
private UserService userService;
@RequestMapping("/user/register")
public String registerUser(){
User user = new User();
user.setPassword("12346");
user.setUsername("admin");
//调用注册业务逻辑
userService.registerUser(user);
return "success";
}
}
5、监听UserRegisterEvent事件
在Spring内部中有多种方式实现监听如:@EventListener注解、实现ApplicationListener泛型接口、实现SmartApplicationListener接口等,我们下面来讲解下这三种方式分别如何实现。
6、@EventListener实现
@Component
public class AnnotationRegisterListener {
@EventListener
public void userRegister(UserRegisterEvent registerEvent){
System.out.println("注册的用户信息如下:"+registerEvent.getUser());
}
@EventListener
public void userRegisterEmail(UserRegisterEvent registerEvent){
System.out.println("用户注册成功"+registerEvent.getUser().getUsername()+",发送邮件");
}
}
我们只需要让我们的监听类被Spring所管理即可,在我们用户注册监听实现方法上添加@EventListener注解,该注解会根据方法内配置的事件完成监听。下面我们启动项目来测试下我们事件发布时是否被监听者所感知。
测试事件监听
浏览器访问http://localhost:8080/user/register,调用接口注册用户,输出信息如下:
注册的用户信息如下:User [username=admin, password=12346]
用户注册成功admin,发送邮件
可以看到我们使用@EventListener注解配置的监听已经生效了,当我们在UserService内发布了注册事件时,监听方法自动被调用并且输出内信息到控制台。
7、ApplicationListener实现监听
在实现ApplicationListener接口时需要将监听事件作为泛型传递,监听实现代码
@Component
public class RegisterApplicationListener implements ApplicationListener<UserRegisterEvent>{
@Override
public void onApplicationEvent(UserRegisterEvent event) {
System.out.println("用户注册信息如下:"+event.getUser());
}
}
@Component
public class RegisterEmailApplicationListener implements ApplicationListener<UserRegisterEvent>{
@Override
public void onApplicationEvent(UserRegisterEvent event) {
System.out.println("用户注册成功,发送邮件");
}
}
我们实现接口后需要使用@Component注解来声明该监听需要被Spring注入管理,当有UserRegisterEvent事件发布时监听程序会自动调用onApplicationEvent方法并且将UserRegisterEvent对象作为参数传递。
测试成功,同步骤6测试
8、SmartApplicationListener实现监听
当多个listener监听同一个事件时,监听是无序的,监听到的事件先后顺序完全随机出现的,通过SmartApplicationListener可以实现监听逻辑的有序性。
@Component
public class RegisterSmartApplicationListener implements SmartApplicationListener{
//supportsEventType和supportsSourceType返回为true时,才会执行监听
@Override
public void onApplicationEvent(ApplicationEvent event) {
System.out.println("用户注册信息如下:"+((UserRegisterEvent)event).getUser());
}
/**
* 同步情况下的事件监听顺序
* @see org.springframework.core.Ordered#getOrder()
*/
@Override
public int getOrder() {
//值越小,优先级越高,执行越靠前
return 0;
}
@Override
public boolean supportsEventType(Class<? extends ApplicationEvent> eventType) {
return eventType == UserRegisterEvent.class;
}
@Override
public boolean supportsSourceType(Class<?> sourceType) {
return sourceType == UserService.class;
}
}
@Component
public class RegisterEmainSmartApplicationListener implements SmartApplicationListener{
//supportsEventType和supportsSourceType返回为true时,才会执行监听
@Override
public void onApplicationEvent(ApplicationEvent event) {
System.out.println("用户注册成功,发送电子邮件"+((UserRegisterEvent)event).getUser());
}
/**
* 同步情况下的事件监听顺序
* @see org.springframework.core.Ordered#getOrder()
*/
@Override
public int getOrder() {
//值越小,优先级越高,执行越靠前,用户注册成功后才发送邮件
return 1;
}
@Override
public boolean supportsEventType(Class<? extends ApplicationEvent> eventType) {
return eventType == UserRegisterEvent.class;
}
@Override
public boolean supportsSourceType(Class<?> sourceType) {
return sourceType == UserService.class;
}
}
通过SmartApplicationListener提供的getOrder方法,该方法可以实现监听的有序性,return的数值越小证明优先级越高,执行顺序越靠前。
测试同步骤6的测试
9、使用@Async实现异步监听
@Aysnc其实是Spring内的一个组件,可以完成对类内单个或者多个方法实现异步调用,这样可以大大的节省等待耗时。内部实现机制是线程池任务ThreadPoolTaskExecutor,通过线程池来对配置@Async的方法或者类做出执行动作。
配置线程池,如下:
@Configuration
@EnableAsync
public class AsyncConfig {
@Bean
public AsyncTaskExecutor getAsyncExecutor() {
ThreadPoolTaskExecutor threadPool = new ThreadPoolTaskExecutor();
threadPool.setCorePoolSize(2);//当前线程数
threadPool.setMaxPoolSize(50);// 最大线程数
threadPool.setQueueCapacity(100);//线程池所使用的缓冲队列
threadPool.setWaitForTasksToCompleteOnShutdown(true);//等待任务在关机时完成--表明等待所有线程执行完
threadPool.setThreadNamePrefix("provider-demo-");
threadPool.setRejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy());
return threadPool;
}
}
使用@Async实现异步监听,在具体的监听方法onApplicationEvent上面添加Async注解,用法同普通的方法上面加Async注解一致
@Override
@Async
public void onApplicationEvent(ApplicationEvent applicationEvent) {
System.out.println("用户注册成功,发送邮件通知。");
}
注意:如果存在多个监听同一个事件时,并且存在异步与同步同时存在时则不存在执行顺序。