1.监听器 Listener 简介
1.1 监听器 Listener 介绍
Listener是JavaWeb的三大组件(Servlet、Filter、Listener)之一,JavaWeb中的监听器主要用于监听:ServletContext、HttpSession、ServletRequest 的生命周期以及属性变化。变量的创建、销毁和修改等。可以在某些动作前后增加处理,实现监控。其工作流程如下图:
1.2 SpringBoot系统监听器介绍
SpringBoot 2.x中定义的事件类型有如下几个:
EventType |
发布时间 |
ApplicationContextInitializedEvent |
SpringApplication正在启动, ApplicationContext 已经准备好了,ApplicationContextInitializers 被调用, bean definitions 被加载之前 |
ApplicationEnvironmentPreparedEvent |
装配完参数和环境后触发的事件 |
ApplicationFailedEvent |
Spring Boot 启动失败时触发 |
ApplicationPreparedEvent |
上下文 Context 准备时触发 |
ApplicationReadyEvent |
上下文准备完毕的时触发 |
ApplicationStartedEvent |
Spring Boot 启动监听类 |
ApplicationStartingEvent |
上下文准备完毕的时触发 |
SpringBoot 2.x事件类图如下:
2.SpringBoot中监听器 Listener 的使用
2.1 监听Servlet上下文对象
监听 Servlet 上下文对象可以用来初始化数据,用于缓存。
示例代码如下,从代码我们可以看出,这个监听类主要是在Spring容器启动之后,自动触发此监听器的onApplicationEvent方法,并执行方法中的内容。:
@Slf4j
@Component
public class MyServletContextListener implements ApplicationListener<ContextRefreshedEvent> {
@Override
public void onApplicationEvent(ContextRefreshedEvent contextRefreshedEvent) {
log.info("Spring容器加载完成触发,可用于初始化环境,准备测试数据、加载一些数据到内存");
//此处可以开启一个线程,用于查询数据,并进行缓存。
//代码省略
}
}
测试结果如下:
我们可以看到,启动过程中加载应用上下文时执行我们编写的监听器代码。在实际开发中,我们可以在次数处理需要预加载或者需要缓存的数据。
2.2 监听 HTTP 会话 Session 对象
常用来监听 Session 对象,以获取在线用户数量
示例代码如下,从代码可以看出该监听器首先需要实现 HttpSessionListener 接口,然后重写 sessionCreated创建方法和 sessionDestroyed 销毁方法,在 sessionCreated 方法中传递一个HttpSessionEvent对象,之后将当前 Session中的用户数量加 1,sessionDestroyed 方法刚好相反。
/**
* 功能描述: HTTP 会话监听器
* @author TuYong
* @date 2022/6/14 10:21
*/
@Component
@Slf4j
public class MyHttpSessionListener implements HttpSessionListener {
public static AtomicInteger count = new AtomicInteger();
@Override
public void sessionCreated(HttpSessionEvent httpSessionEvent) {
log.info("新用户上线了");
count.incrementAndGet();
httpSessionEvent.getSession().getServletContext().setAttribute("count", count.get());
}
@Override
public void sessionDestroyed(HttpSessionEvent httpSessionEvent) {
log.info("用户下线了");
count.decrementAndGet();
httpSessionEvent.getSession().getServletContext().setAttribute("count", count.get());
}
}
@RestController
@RequestMapping("/event")
public class EventController {
@GetMapping("/test")
public Object test(HttpServletRequest request){
Integer count = (Integer) request.getSession().getServletContext().getAttribute("count");
return "当前在线人数:" + count;
}
}
我们在不同浏览器打开 http://127.0.0.1:8080/event/test,可以看到如下结果。
2.3 监听客户端请求 Servlet Request 对象
示例代码如下,实现ServletRequestListener接口即可,然后通过request对象获取一些信息。
/**
* 功能描述: 客户端请求 Servlet Request 监听
* @author TuYong
* @date 2022/6/14 10:44
*/
@Component
@Slf4j
public class MyServletRequestListener implements ServletRequestListener {
@Override
public void requestInitialized(ServletRequestEvent servletRequestEvent) {
HttpServletRequest request = (HttpServletRequest) servletRequestEvent.getServletRequest();
log.info("session id:{}", request.getRequestedSessionId());
log.info("request url:{}", request.getRequestURL());
request.setAttribute("username", "Trazen");
}
@Override
public void requestDestroyed(ServletRequestEvent servletRequestEvent) {
log.info("request end");
HttpServletRequest request = (HttpServletRequest) servletRequestEvent.getServletRequest();
log.info("request域中保存的username值为{}", request.getAttribute("username"));
}
}
我们请求接口http://127.0.0.1:8080/event/test,可以看到控制台打印如下:
2.4 自定义事件监听器
在实际项目中,我们通常需要自定义一些事件和监听器来满足业务场景,比如在微服务中会有这样的场景:微服务A在处理完某个逻辑之后,需要通知微服务B去处理另一个逻辑,或者微服务 A 处理完某个逻辑之后,需要将数据同步到微服务 B,这种场景非常普遍,这个时候我们可以自定义事件以及监听器来监听,一旦监听到微服务A 中的某事件发生,就去通知微服务 B 处理对应的逻辑。
自定义事件监听器步骤如下:
- 自定义事件:编写一个事件,只需要继承ApplicationEvent即可
- 自定义监听器:自定义监听器实现 ApplicationListener接口,把自定义的事件作为参数,重写onApplicationEvent方法即可
- 触发器:注入ApplicationContext,调用publishEvent发布自定义事件即可
- 调用触发器:简单了,就是写个controller类,调用触发器方法即可
2.4.1 自定义事件
编写一个事件,只需要继承 ApplicationEvent对象,在事件中定义一个User对象来模拟数据,构造方法中将 User对象传进来初始化。如下:
/**
* 功能描述: 自定义事件
* @author TuYong
* @date 2022/6/14 11:11
*/
public class MyEvent extends ApplicationEvent {
@Getter
@Setter
private User user;
public MyEvent(Object source, User user) {
super(source);
this.user = user;
}
}
@Data
public class User {
private String username;
}
2.4.2 自定义监听器
自定义一个监听器来监听上面定义的MyEvent事件,自定义监听器需要实现 ApplicationListener接口即可。
具体代码如下,重写onApplicationEvent方法,将自定义的MyEvent事件传进来,因为该事件中,我们定义了 User对象,然后就可以使用该对象的信息了。
/**
* 功能描述: 自定义监听器
* @author TuYong
* @date 2022/6/14 11:19
*/
@Component
public class MyEventListener implements ApplicationListener<MyEvent> {
@Override
public void onApplicationEvent(MyEvent myEvent) {
// 把事件中的信息获取到
User user = myEvent.getUser();
// 处理事件,实际项目中可以处理其他逻辑等等
System.out.println("用户名:" + user.getUsername());
}
}
2.4.3 发布事件
代码如下,我们在service中注入ApplicationContext,在业务代码处理完之后,通过ApplicationContext对象手动发布 MyEvent事件,这样我们自定义的监听器就能监听到,然后处理监听器中写好的业务逻辑。
public interface UserService {
/**
* add user
*/
void addUser();
}
@Service
@RequiredArgsConstructor
public class UserServiceImpl implements UserService {
private final ApplicationContext applicationContext;
@Override
public void addUser() {
User user = new User();
user.setUsername("Trazen");
//发布事件
MyEvent event = new MyEvent(this, user);
applicationContext.publishEvent(event);
}
}
@RestController
@RequestMapping("/event")
@RequiredArgsConstructor
public class EventController {
private final UserService userService;
@GetMapping("/addUser")
public Object addUser(){
userService.addUser();
return "success";
}
}
测试结果如下:
我们请求链接http://127.0.0.1:8080/event/addUser,可以看到监听器生效。
免责声明:本站所有文章内容,图片,视频等均是来源于用户投稿和互联网及文摘转载整编而成,不代表本站观点,不承担相关法律责任。其著作权各归其原作者或其出版社所有。如发现本站有涉嫌抄袭侵权/违法违规的内容,侵犯到您的权益,请在线联系站长,一经查实,本站将立刻删除。 本文来自网络,若有侵权,请联系删除,如若转载,请注明出处:https://yundeesoft.com/83715.html