微服务——SpringBoot使用归纳——Spring Boot中使用监听器——监听器介绍和使用

简介: 本文介绍了在Spring Boot中使用监听器的方法。首先讲解了Web监听器的概念,即通过监听特定事件(如ServletContext、HttpSession和ServletRequest的创建与销毁)实现监控和处理逻辑。接着详细说明了三种实际应用场景:1) 监听Servlet上下文对象以初始化缓存数据;2) 监听HTTP会话Session对象统计在线用户数;3) 监听客户端请求的Servlet Request对象获取访问信息。每种场景均配有代码示例,帮助开发者理解并应用监听器功能。

Spring Boot中使用监听器

1. 监听器介绍

什么是 web 监听器?web 监听器是一种 Servlet 中特殊的类,它们能帮助开发者监听 web 中特定的事件,比如 ServletContext, HttpSession, ServletRequest 的创建和销毁;变量的创建、销毁和修改等。可以在某些动作前后增加处理,实现监控。

2. Spring Boot中监听器的使用

web 监听器的使用场景很多,比如监听 servlet 上下文用来初始化一些数据、监听 http session 用来获取当前在线的人数、监听客户端请求的 servlet request 对象来获取用户的访问信息等等。这一节中,我们主要通过这三个实际的使用场景来学习一下 Spring Boot 中监听器的使用。

2.1 监听Servlet上下文对象

监听 servlet 上下文对象可以用来初始化数据,用于缓存。什么意思呢?我举一个很常见的场景,比如用户在点击某个站点的首页时,一般都会展现出首页的一些信息,而这些信息基本上或者大部分时间都保持不变的,但是这些信息都是来自数据库。如果用户的每次点击,都要从数据库中去获取数据的话,用户量少还可以接受,如果用户量非常大的话,这对数据库也是一笔很大的开销。  

针对这种首页数据,大部分都不常更新的话,我们完全可以把它们缓存起来,每次用户点击的时候,我们都直接从缓存中拿,这样既可以提高首页的访问速度,又可以降低服务器的压力。如果做的更加灵活一点,可以再加个定时器,定期的来更新这个首页缓存。就类似与 CSDN 个人博客首页中排名的变化一样。

下面我们针对这个功能,来写一个 demo,在实际中,读者可以完全套用该代码,来实现自己项目中的相关逻辑。首先写一个 Service,模拟一下从数据库查询数据:

@Service

public class UserService {


   /**

    * 获取用户信息

    * @return

    */

   public User getUser() {

       // 实际中会根据具体的业务场景,从数据库中查询对应的信息

       return new User(1L, "倪升武", "123456");

   }

}

然后写一个监听器,实现 ApplicationListener<ContextRefreshedEvent> 接口,重写 onApplicationEvent 方法,将 ContextRefreshedEvent 对象传进去。如果我们想在加载或刷新应用上下文时,也重新刷新下我们预加载的资源,就可以通过监听 ContextRefreshedEvent 来做这样的事情。如下:

/**

* 使用ApplicationListener来初始化一些数据到application域中的监听器

* @author shengni ni

* @date 2018/07/05

*/

@Component

public class MyServletContextListener implements ApplicationListener<ContextRefreshedEvent> {


   @Override

   public void onApplicationEvent(ContextRefreshedEvent contextRefreshedEvent) {

       // 先获取到application上下文

       ApplicationContext applicationContext = contextRefreshedEvent.getApplicationContext();

       // 获取对应的service

       UserService userService = applicationContext.getBean(UserService.class);

       User user = userService.getUser();

       // 获取application域对象,将查到的信息放到application域中

       ServletContext application = applicationContext.getBean(ServletContext.class);

       application.setAttribute("user", user);

   }

}

正如注释中描述的一样,首先通过 contextRefreshedEvent 来获取 application 上下文,再通过 application 上下文来获取 UserService 这个 bean,项目中可以根据实际业务场景,也可以获取其他的 bean,然后再调用自己的业务代码获取相应的数据,最后存储到 application 域中,这样前端在请求相应数据的时候,我们就可以直接从 application 域中获取信息,减少数据库的压力。下面写一个 Controller 直接从 application 域中获取 user 信息来测试一下。

@RestController

@RequestMapping("/listener")

public class TestController {


   @GetMapping("/user")

   public User getUser(HttpServletRequest request) {

       ServletContext application = request.getServletContext();

       return (User) application.getAttribute("user");

   }

}

启动项目,在浏览器中输入 http://localhost:8080/listener/user 测试一下即可,如果正常返回 user 信息,那么说明数据已经缓存成功。不过 application 这种是缓存在内存中,对内存会有消耗,后面的课程中我会讲到 redis,到时候再给大家介绍一下 redis 的缓存。

2.2 监听HTTP会话 Session对象

监听器还有一个比较常用的地方就是用来监听 session 对象,来获取在线用户数量,现在有很多开发者都有自己的网站,监听 session 来获取当前在下用户数量是个很常见的使用场景,下面来介绍一下如何来使用。

/**

* 使用HttpSessionListener统计在线用户数的监听器

* @author shengwu ni

* @date 2018/07/05

*/

@Component

public class MyHttpSessionListener implements HttpSessionListener {


   private static final Logger logger = LoggerFactory.getLogger(MyHttpSessionListener.class);


   /**

    * 记录在线的用户数量

    */

   public Integer count = 0;


   @Override

   public synchronized void sessionCreated(HttpSessionEvent httpSessionEvent) {

       logger.info("新用户上线了");

       count++;

       httpSessionEvent.getSession().getServletContext().setAttribute("count", count);

   }


   @Override

   public synchronized void sessionDestroyed(HttpSessionEvent httpSessionEvent) {

       logger.info("用户下线了");

       count--;

       httpSessionEvent.getSession().getServletContext().setAttribute("count", count);

   }

}

可以看出,首先该监听器需要实现 HttpSessionListener 接口,然后重写 sessionCreatedsessionDestroyed 方法,在 sessionCreated 方法中传递一个 HttpSessionEvent 对象,然后将当前 session 中的用户数量加1,sessionDestroyed 方法刚好相反,不再赘述。然后我们写一个 Controller 来测试一下。

@RestController

@RequestMapping("/listener")

public class TestController {


   /**

    * 获取当前在线人数,该方法有bug

    * @param request

    * @return

    */

   @GetMapping("/total")

   public String getTotalUser(HttpServletRequest request) {

       Integer count = (Integer) request.getSession().getServletContext().getAttribute("count");

       return "当前在线人数:" + count;

   }

}

该 Controller 中是直接获取当前 session 中的用户数量,启动服务器,在浏览器中输入 localhost:8080/listener/total 可以看到返回的结果是1,再打开一个浏览器,请求相同的地址可以看到 count 是 2 ,这没有问题。但是如果关闭一个浏览器再打开,理论上应该还是2,但是实际测试却是 3。原因是 session 销毁的方法没有执行(可以在后台控制台观察日志打印情况),当重新打开时,服务器找不到用户原来的 session,于是又重新创建了一个 session,那怎么解决该问题呢?我们可以将上面的 Controller 方法改造一下:

@GetMapping("/total2")

public String getTotalUser(HttpServletRequest request, HttpServletResponse response) {

   Cookie cookie;

   try {

       // 把sessionId记录在浏览器中

       cookie = new Cookie("JSESSIONID", URLEncoder.encode(request.getSession().getId(), "utf-8"));

       cookie.setPath("/");

       //设置cookie有效期为2天,设置长一点

       cookie.setMaxAge( 48*60 * 60);

       response.addCookie(cookie);

   } catch (UnsupportedEncodingException e) {

       e.printStackTrace();

   }

   Integer count = (Integer) request.getSession().getServletContext().getAttribute("count");

   return "当前在线人数:" + count;

}

可以看出,该处理逻辑是让服务器记得原来那个 session,即把原来的 sessionId 记录在浏览器中,下次再打开时,把这个 sessionId 传过去,这样服务器就不会重新再创建了。重启一下服务器,在浏览器中再次测试一下,即可避免上面的问题。

2.3 监听客户端请求Servlet Request对象

使用监听器获取用户的访问信息比较简单,实现 ServletRequestListener 接口即可,然后通过 request 对象获取一些信息。如下:

/**

* 使用ServletRequestListener获取访问信息

* @author shengwu ni

* @date 2018/07/05

*/

@Component

public class MyServletRequestListener implements ServletRequestListener {


   private static final Logger logger = LoggerFactory.getLogger(MyServletRequestListener.class);


   @Override

   public void requestInitialized(ServletRequestEvent servletRequestEvent) {

       HttpServletRequest request = (HttpServletRequest) servletRequestEvent.getServletRequest();

       logger.info("session id为:{}", request.getRequestedSessionId());

       logger.info("request url为:{}", request.getRequestURL());


       request.setAttribute("name", "倪升武");

   }


   @Override

   public void requestDestroyed(ServletRequestEvent servletRequestEvent) {


       logger.info("request end");

       HttpServletRequest request = (HttpServletRequest) servletRequestEvent.getServletRequest();

       logger.info("request域中保存的name值为:{}", request.getAttribute("name"));


   }


}

这个比较简单,不再赘述,接下来写一个 Controller 测试一下即可。

@GetMapping("/request")

public String getRequestInfo(HttpServletRequest request) {

   System.out.println("requestListener中的初始化的name数据:" + request.getAttribute("name"));

   return "success";

}

目录
相关文章
|
8月前
|
负载均衡 Java API
《深入理解Spring》Spring Cloud 构建分布式系统的微服务全家桶
Spring Cloud为微服务架构提供一站式解决方案,涵盖服务注册、配置管理、负载均衡、熔断限流等核心功能,助力开发者构建高可用、易扩展的分布式系统,并持续向云原生演进。
|
8月前
|
XML Java 应用服务中间件
【SpringBoot(一)】Spring的认知、容器功能讲解与自动装配原理的入门,带你熟悉Springboot中基本的注解使用
SpringBoot专栏开篇第一章,讲述认识SpringBoot、Bean容器功能的讲解、自动装配原理的入门,还有其他常用的Springboot注解!如果想要了解SpringBoot,那么就进来看看吧!
738 2
|
9月前
|
监控 Java 数据库
从零学 Dropwizard:手把手搭轻量 Java 微服务,告别 Spring 臃肿
Dropwizard 整合 Jetty、Jersey 等成熟组件,开箱即用,无需复杂配置。轻量高效,启动快,资源占用少,内置监控、健康检查与安全防护,搭配 Docker 部署便捷,是构建生产级 Java 微服务的极简利器。
927 117
|
9月前
|
人工智能 Java 机器人
基于Spring AI Alibaba + Spring Boot + Ollama搭建本地AI对话机器人API
Spring AI Alibaba集成Ollama,基于Java构建本地大模型应用,支持流式对话、knife4j接口可视化,实现高隐私、免API密钥的离线AI服务。
7028 2
基于Spring AI Alibaba + Spring Boot + Ollama搭建本地AI对话机器人API
存储 JSON Java
927 0
|
10月前
|
监控 Java API
Spring Boot 3.2 结合 Spring Cloud 微服务架构实操指南 现代分布式应用系统构建实战教程
Spring Boot 3.2 + Spring Cloud 2023.0 微服务架构实践摘要 本文基于Spring Boot 3.2.5和Spring Cloud 2023.0.1最新稳定版本,演示现代微服务架构的构建过程。主要内容包括: 技术栈选择:采用Spring Cloud Netflix Eureka 4.1.0作为服务注册中心,Resilience4j 2.1.0替代Hystrix实现熔断机制,配合OpenFeign和Gateway等组件。 核心实操步骤: 搭建Eureka注册中心服务 构建商品
1410 3
|
人工智能 安全 Java
微服务引擎 MSE:打造通用的企业级微服务架构
微服务引擎MSE致力于打造通用的企业级微服务架构,涵盖四大核心内容:微服务技术趋势与挑战、MSE应对方案、拥抱开源及最佳实践。MSE通过流量入口、内部流量管理、服务治理等模块,提供高可用、跨语言支持和性能优化。此外,MSE坚持开放,推动云原生与AI融合,助力企业实现无缝迁移和高效运维。
781 1