【JavaWeb】Servlet监听器
前言
本文为JavaWeb基础Servlet监听器相关知识介绍~, Java全栈学习路线可参考:【Java全栈学习路线】最全的Java学习路线及知识清单,Java自学方向指引,内含最全Java全栈学习技术清单~
一、监听器介绍
- 概述: Listener是Servlet的监听器,它可以监听客户端的请求、服务端的操作等。通过监听器,可以自动激发一些操作,比如监听在线的用户的数量。监听的事件源分别为SerlvetConext,HttpSession和ServletRequest这三个域对象。
- 作用: 监听器可以监听 Web 中特定的事件。
- 方式: 监听器监听的是拥有作用域的对象,如 ServletContext、HttpSession 等。这些作用域对象分别都有相同的方法,如setAttribute、getAttribute、removeAttribute,只要调用这几个方法就会进入到监听器对应的方法中处理相应的逻辑。
- 场景: 初始化上下文(Spring 容器的初始化,文件的解析)、会话的监听(在线人数),监听干预用户信息(用户资源倾斜)。
二、监听application
(1)定义一个监听器实现 ApplicationListener 接口,重写 onApplicationEvent 方法,获取 applicationContext 上下文中的 Bean 对象,执行相应方法后将结果放到 ServletContext 的作用域对象 application 中。
@Component
public class ServletContextListener implements ApplicationListener<ContextRefreshedEvent> {
@Override
public void onApplicationEvent(ContextRefreshedEvent event) {
if (event.getApplicationContext().getParent() == null) {
ApplicationContext applicationContext = event.getApplicationContext();
UserService userService = applicationContext.getBean(UserService.class);
User user = userService.getUser();
ServletContext application = applicationContext.getBean(ServletContext.class);
application.setAttribute("user", user);
}
}
}
(2)定义测试接口
@RestController
@Slf4j
public class UserController {
@GetMapping("/api/getUser")
public User getUser(HttpServletRequest request) {
ServletContext servletContext = request.getServletContext();
User user = (User) servletContext.getAttribute("user");
log.info("user: {}", user);
return user;
}
}
(3)查看测试结果
2021-12-28 10:55:56.472 INFO 10820 --- [nio-8080-exec-1] com.xpy.controller.UserController : user: User(id=1, nickname=PlanMak1r, password=123456)
三、监听session
(1)定义一个监听器实现 HttpSessionListener 接口,重写两个方法,获取 session 会话对象,并将数据放到 HttpSession 的作用域对象 session 中
@Component
@Slf4j
public class LoginSessionListener implements HttpSessionListener {
public static Integer count = 0;
@Override
public void sessionCreated(HttpSessionEvent se) {
log.info("新用户上线...");
count++;
se.getSession().getServletContext().setAttribute("personCount", count);
}
@Override
public void sessionDestroyed(HttpSessionEvent se) {
log.info("用户下线了...");
count--;
se.getSession().getServletContext().setAttribute("personCount", count);
}
}
(2)定义测试接口
@RestController
@Slf4j
public class LoginController {
@GetMapping("/getOnlineUser")
public String getOnlineUser(HttpServletRequest request) {
Integer personCount = (Integer) request.getServletContext().getAttribute("personCount");
log.info("登录的人数是: {}", (personCount == null ? 0 : personCount));
return "登录的人数是:" + (personCount == null ? 0 : personCount);
}
@GetMapping("logined")
public String logined(HttpSession session) {
session.setAttribute("user", new User(1L, "PlanMak1r", "123456"));
log.info("登录成功...");
return "success";
}
@GetMapping("logout")
public String logout(HttpSession session) {
session.invalidate();
log.info("退出成功...");
return "success";
}
}
(3)查看测试结果
2021-12-28 11:28:23.420 INFO 8232 --- [nio-8080-exec-1] com.xpy.controller.LoginController : 登录的人数是: 0
2021-12-28 11:28:29.436 INFO 8232 --- [nio-8080-exec-2] c.x.c.listener.LoginSessionListener : 新用户上线...
2021-12-28 11:28:29.450 INFO 8232 --- [nio-8080-exec-2] com.xpy.controller.LoginController : 登录成功...
2021-12-28 11:28:39.832 INFO 8232 --- [nio-8080-exec-3] com.xpy.controller.LoginController : 登录的人数是: 1
2021-12-28 11:28:49.381 INFO 8232 --- [nio-8080-exec-5] c.x.c.listener.LoginSessionListener : 用户下线了...
2021-12-28 11:28:49.381 INFO 8232 --- [nio-8080-exec-5] com.xpy.controller.LoginController : 退出成功...
2021-12-28 11:28:51.167 INFO 8232 --- [nio-8080-exec-6] com.xpy.controller.LoginController : 登录的人数是: 0
(4)导入 freemarker 依赖
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-freemarker</artifactId>
</dependency>
(5)配置 freemarker
spring:
freemarker:
suffix: .html
cache: false
(6)定义页面跳转
@Controller
public class IndexController {
@GetMapping("/index")
public String index() {
return "index";
}
}
(7)定义展示页面实现实时展示
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>首页</title>
</head>
<body>
<h1><span id="personCount"></span></h1>
<script src="/js/jquery.min.js"></script>
<script>
$(function () {
function loadOnlineUserCount() {
$.get("/getOnlineUser", function (res) {
$("#personCount").html(res);
});
}
loadOnlineUserCount();
setInterval(loadOnlineUserCount, 3000);
})
</script>
</body>
</html>
四、监听request
对request对象实际上就是对ServletRequest进行监听,主要的监听接口有两个:
- ServletRequestListener : 对浏览器的请求的创建和销毁进行监听
- ServletRequestAttributeListener :对request域对象范围属性的设置、移除和修改进行监听
实现ServletRequestListener接口,对浏览器的请求的创建和销毁进行监听,复写的方法有两个:
- void requestInitialized(ServletRequestEvent sre):请求的创建
- void requestDestroyed(ServletRequestEvent sre):请求的销毁
实现ServletRequestAttributeListener接口,对request域对象范围属性的设置、移除和修改进行监听,复写的方法有三个:
- void attributeAdded(ServletRequestAttributeEvent srae):将信息添加到request域对象时执行
- void attributeRemoved(ServletRequestAttributeEvent srae):从request域对象中移除信息时执行
- void attributeReplaced(ServletRequestAttributeEvent srae):修改request域对象中的信息时执行
package com.wanbangee.listener;
import javax.servlet.ServletRequestAttributeEvent;
import javax.servlet.ServletRequestAttributeListener;
import javax.servlet.ServletRequestEvent;
import javax.servlet.ServletRequestListener;
import javax.servlet.annotation.WebListener;
@WebListener
public class ServletRequestListenerDemo implements ServletRequestListener, ServletRequestAttributeListener {
public void requestInitialized(ServletRequestEvent sre) {
System.out.println("请求的创建");
}
public void requestDestroyed(ServletRequestEvent sre) {
System.out.println("请求的销毁");
}
public void attributeAdded(ServletRequestAttributeEvent srae) {
System.out.println("将信息添加到request域对象时执行");
}
public void attributeRemoved(ServletRequestAttributeEvent srae) {
System.out.println("从request域对象中移除信息时执行");
}
public void attributeReplaced(ServletRequestAttributeEvent srae) {
System.out.println("修改request域对象中的信息时执行");
}
}
后记
总结:
- 使用监听器可以对application、session和request的范围属性进行监听
- 使用ServletContextListener可以监听服务器的启动和关闭(application对象)
- 使用HttpSessionListener可以监听session的创建和session的注销(session对象)
- 使用ServletRequestListener可以监听请求的创建和销毁(request对象)
- 监听器的配置主要是使用注解的方式完成
- 一个监听器可以同时实现多个监听接口,因为接口是多实现的(接口多继承多实现)
Java全栈学习路线可参考:【Java全栈学习路线】最全的Java学习路线及知识清单,Java自学方向指引,内含最全Java全栈学习技术清单~
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。
文章由极客之音整理,本文链接:https://www.bmabk.com/index.php/post/154186.html