拦截器

流程图

实现HandlerInterceptor

位置

interceptor目录下

具体流程

  • 当我们打开浏览器来访问部署在web服务器当中的web应用时,此时我们所定义的过滤器会拦截到这次请求。拦截到这次请求之后,它会先执行放行前的逻辑,然后再执行放行操作。而由于我们当前是基于springboot开发的,所以放行之后是进入到了spring的环境当中,也就是要来访问我们所定义的controller当中的接口方法。

  • Tomcat并不识别所编写的Controller程序,但是它识别Servlet程序,所以在Spring的Web环境中提供了一个非常核心的Servlet:DispatcherServlet(前端控制器),所有请求都会先进行到DispatcherServlet,再将请求转给Controller。

  • 当我们定义了拦截器后,拦截器会在执行Controller的方法之前拦截住请求,执行preHandle()方法,这个方法执行完成后需要返回一个布尔类型的值,如果返回true,就表示放行本次操作,才会继续访问controller中的方法;如果返回false,则不会放行(controller中的方法也不会执行)。

  • 在controller当中的方法执行完毕之后,再回过来执行postHandle()这个方法以及afterCompletion() 方法,然后再返回给DispatcherServlet,最终再来执行过滤器当中放行后的这一部分逻辑的逻辑。执行完毕之后,最终给浏览器响应数据。

简略流程

  1. preHandle : 调用controller之前
  2. postHandle:调用controller之后
  3. afterCompletion:视图渲染之后(这里的视图渲染指的是服务器将动态资源转化为静态资源)

核心代码

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
@Component
public class LoginCheckInterceptor implements HandlerInterceptor {
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
System.out.println("preHandle .... ");

return true; //true表示放行
}

@Override
public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
System.out.println("postHandle ... ");
}

@Override
public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
System.out.println("afterCompletion .... ");
}
}

配置拦截器

位置

config目录下

简略流程

  1. 实现 WebMvcConfigurer
  2. 重写 addInterceptors

核心代码

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
// 拦截器关键代码
@Configuration
public class WebConfig implements WebMvcConfigurer {

// 拦截器对象
@Autowired
private LoginCheckInterceptor loginCheckInterceptor;

@Override
public void addInterceptors(InterceptorRegistry registry) {
// 注册自定义拦截器对象
registry.addInterceptor(loginCheckInterceptor)
.addPathPatterns("/**") //拦截所有请求
.excludePathPatterns("/login"); //不拦截登录请求
}
}

统一异常处理

核心注解

  1. @ControllerAdvice :异常控制器,推荐使用@RestControllerAdvice(整体代码更简洁)
  2. @ExceptionHandler: 声明异常类型

核心代码

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
// @ControllerAdvice
// public class GlobalExceptionHandler {
//
// // 需要手动添加 @ResponseBody 才能返回 JSON
// @ExceptionHandler(Exception.class)
// @ResponseBody
// public Result ex(Exception ex){
// ex.printStackTrace();
// return Result.error("对不起,操作失败,请联系管理员");
// }
// }

@RestControllerAdvice
public class GlobalExceptionHandler {

@ExceptionHandler(Exception.class)//捕获所有异常
public Result ex(Exception ex){
ex.printStackTrace();
return Result.error("对不起,操作失败,请联系管理员");
}

}