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

ApplicationEvent&ApplicationListener使用

程序员文章站 2022-07-01 23:14:50
...

在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("用户注册成功,发送邮件通知。");

    }

 

 

注意:如果存在多个监听同一个事件时,并且存在异步与同步同时存在时则不存在执行顺序。