博客列表 >20、【转载】Spring Boot拦截器精讲

20、【转载】Spring Boot拦截器精讲

自由之上
自由之上原创
2022年11月13日 09:02:07536浏览

我们对拦截器并不陌生,无论是 Struts 2 还是 Spring MVC 中都提供了拦截器功能,它可以根据 URL 对请求进行拦截,主要应用于登陆校验、权限验证、乱码解决、性能监控和异常处理等功能上。Spring Boot 同样提供了拦截器功能。

在 Spring Boot 项目中,使用拦截器功能通常需要以下 3 步:

  1. 定义拦截器;
  2. 注册拦截器;
  3. 指定拦截规则(如果是拦截所有,静态资源也会被拦截)。

1、定义拦截器

在 Spring Boot 中定义拦截器十分的简单,只需要创建一个拦截器类,并实现 HandlerInterceptor 接口即可。
HandlerInterceptor 接口中定义以下 3 个方法,如下表。

返回值类型 方法声明 描述
boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) 该方法在控制器处理请求方法前执行,其返回值表示是否中断后续操作,返回 true 表示继续向下执行,返回 false 表示中断后续操作。
void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) 该方法在控制器处理请求方法调用之后、解析视图之前执行,可以通过此方法对请求域中的模型和视图做进一步修改。
void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) 该方法在视图渲染结束后执行,可以通过此方法实现资源清理、记录日志信息等工作。

示例1
以 spring-boot-adminex 项目为例,在 net.biancheng.www.componet 中创建一个名为 LoginInterceptor 的拦截器类,对登陆进行拦截,代码如下。

  1. package net.biancheng.www.componet;
  2. import lombok.extern.slf4j.Slf4j;
  3. import org.springframework.web.servlet.HandlerInterceptor;
  4. import org.springframework.web.servlet.ModelAndView;
  5. import javax.servlet.http.HttpServletRequest;
  6. import javax.servlet.http.HttpServletResponse;
  7. @Slf4j
  8. public class LoginInterceptor implements HandlerInterceptor {
  9. /**
  10. * 目标方法执行前
  11. *
  12. * @param request
  13. * @param response
  14. * @param handler
  15. * @return
  16. * @throws Exception
  17. */
  18. @Override
  19. public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
  20. Object loginUser = request.getSession().getAttribute("loginUser");
  21. if (loginUser == null) {
  22. //未登录,返回登陆页
  23. request.setAttribute("msg", "您没有权限进行此操作,请先登陆!");
  24. request.getRequestDispatcher("/index.html").forward(request, response);
  25. return false;
  26. } else {
  27. //放行
  28. return true;
  29. }
  30. }
  31. /**
  32. * 目标方法执行后
  33. *
  34. * @param request
  35. * @param response
  36. * @param handler
  37. * @param modelAndView
  38. * @throws Exception
  39. */
  40. @Override
  41. public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
  42. log.info("postHandle执行{}", modelAndView);
  43. }
  44. /**
  45. * 页面渲染后
  46. *
  47. * @param request
  48. * @param response
  49. * @param handler
  50. * @param ex
  51. * @throws Exception
  52. */
  53. @Override
  54. public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
  55. log.info("afterCompletion执行异常{}", ex);
  56. }
  57. }

2、注册拦截器

创建一个实现了 WebMvcConfigurer 接口的配置类(使用了 @Configuration 注解的类),重写 addInterceptors() 方法,并在该方法中调用 registry.addInterceptor() 方法将自定义的拦截器注册到容器中。
示例 2
在配置类 MyMvcConfig 中,添加以下方法注册拦截器,代码如下。

  1. @Configuration
  2. public class MyMvcConfig implements WebMvcConfigurer {
  3. ......
  4. @Override
  5. public void addInterceptors(InterceptorRegistry registry) {
  6. registry.addInterceptor(new LoginInterceptor());
  7. }
  8. }

3、指定拦截规则

修改 MyMvcConfig 配置类中 addInterceptors() 方法的代码,继续指定拦截器的拦截规则,代码如下。

  1. @Slf4j
  2. @Configuration
  3. public class MyMvcConfig implements WebMvcConfigurer {
  4. ......
  5. @Override
  6. public void addInterceptors(InterceptorRegistry registry) {
  7. log.info("注册拦截器");
  8. registry.addInterceptor(new LoginInterceptor())
  9. .addPathPatterns("/**") //拦截所有请求,包括静态资源文件
  10. .excludePathPatterns("/", "/login", "/index.html", "/user/login", "/css/**", "/images/**", "/js/**", "/fonts/**"); //放行登录页,登陆操作,静态资源
  11. }
  12. }

在指定拦截器拦截规则时,调用了两个方法,这两个方法的说明如下:

  • addPathPatterns:该方法用于指定拦截路径,例如拦截路径为“/**”,表示拦截所有请求,包括对静态资源的请求。
  • excludePathPatterns:该方法用于排除拦截路径,即指定不需要被拦截器拦截的请求。

至此,拦截器的基本功能已经完成,接下来,我们先实现 spring-boot-adminex 的登陆功能,为验证登陆拦截做准备。

4、实现登陆功能

1、将 AdminEx 模板中的 index.html 改名为 main.html,并移动到 src/main/resources/templates 中,结构如下图。

图1:main 页面

2、在 net.bianheng.www.controller 中创建一个 LoginController, 并在其中添加处理登陆请求的方法 doLogin(),代码如下。

  1. package net.biancheng.www.controller;
  2. import lombok.extern.slf4j.Slf4j;
  3. import net.biancheng.www.bean.User;
  4. import org.springframework.stereotype.Controller;
  5. import org.springframework.util.StringUtils;
  6. import org.springframework.web.bind.annotation.GetMapping;
  7. import org.springframework.web.bind.annotation.PostMapping;
  8. import org.springframework.web.bind.annotation.RequestMapping;
  9. import javax.servlet.http.HttpSession;
  10. import java.util.Map;
  11. @Slf4j
  12. @Controller
  13. public class LoginController {
  14. @RequestMapping("/user/login")
  15. public String doLogin(User user, Map<String, Object> map, HttpSession session) {
  16. if (user != null && StringUtils.hasText(user.getUsername()) && "123456".equals(user.getPassword())) {
  17. session.setAttribute("loginUser", user);
  18. log.info("登陆成功,用户名:" + user.getUsername());
  19. //防止重复提交使用重定向
  20. return "redirect:/main.html";
  21. } else {
  22. map.put("msg", "用户名或密码错误");
  23. log.error("登陆失败");
  24. return "login";
  25. }
  26. }
  27. /*
  28. @RequestMapping("/main.html")
  29. public String mainPage(){
  30. return "main";
  31. }*/
  32. }

3、 在配置类 MyMvcConfig 中添加视图映射,代码如下。

  1. @Configuration
  2. public class MyMvcConfig implements WebMvcConfigurer {
  3. @Override
  4. public void addViewControllers(ViewControllerRegistry registry) {
  5. //当访问 “/” 或 “/index.html” 时,都直接跳转到登陆页面
  6. registry.addViewController("/").setViewName("login");
  7. registry.addViewController("/index.html").setViewName("login");
  8. //添加视图映射 main.html 指向 dashboard.html
  9. registry.addViewController("/main.html").setViewName("main");
  10. }
  11. ......
  12. }

4、在 login.html 适当位置添加以下代码,显示错误信息。

  1. <p style="color: red" th:text="${msg}" th:if="${not #strings.isEmpty(msg)}"></p>

5、验证登陆及登陆拦截功能
1、 启动 Spring Boot,在未登录的情况下,直接通过“http://localhost:8080/main.html”访问主页,结果如下图。

图1:登陆被拦截

2、在登陆页用户名和密码输入框内分别输入 “admin”和“admin123”,点击下方的登陆按钮,结果如下图。

图2:登陆失败

3、在登陆页用户名和密码输入框内分别输入 “admin”和“123456”,点击下方的登陆按钮,结果如下图。

图3:登陆成功

声明:本文内容转载自脚本之家,由网友自发贡献,版权归原作者所有,如您发现涉嫌抄袭侵权,请联系admin@php.cn 核实处理。
全部评论
文明上网理性发言,请遵守新闻评论服务协议