生活不易,且行且学习吧
转载 https://blog.youkuaiyun.com/eson_15/article/details/104347476
拦截器的原理很简单,是
AOP
的一种实现,专门拦截对动态资源的后台请求,即拦截对控制层的请 求。使用场景比较多的是判断用户是否有权限请求后台,更拔高一层的使用场景也有,比如拦截器可以 结合 websocket
一起使用,用来拦截
websocket
请求,然后做相应的处理等等。拦截器不会拦截静态 资源,Spring Boot
的默认静态目录为
resources/static
,该目录下的静态页面、
js
、
css
、图片等等, 不会被拦截(也要看如何实现,有些情况也会拦截,我在下文会指出)。
1.
定义拦截器
定义拦截器,只需要实现
HandlerInterceptor
接口,
HandlerInterceptor
接口是所有自定义拦截 器或者 Spring Boot
提供的拦截器的鼻祖,所以,首先来了解下该接口。该接口中有三个方法: preHandle(……) 、
postHandle(……)
和
afterCompletion(……)
。
preHandle(……)
方法:该方法的执行时机是,当某个
url
已经匹配到对应的
Controller
中的某 个方法,且在这个方法执行之前。所以 preHandle(……)
方法可以决定是否将请求放行,这是通 过返回值来决定的,返回 true
则放行,返回
false
则不会向后执行。
postHandle(……)
方法:该方法的执行时机是,当某个
url
已经匹配到对应的
Controller
中的某 个方法,且在执行完了该方法,但是在 DispatcherServlet
视图渲染之前。所以在这个方法中有 个 ModelAndView
参数,可以在此做一些修改动作。
afterCompletion(……)
方法:顾名思义,该方法是在整个请求处理完成后(包括视图渲染)执 行,这时做一些资源的清理工作,这个方法只有在 preHandle(……)
被成功执行后并且返回
true 才会被执行。
2.
配置拦截器
在
Spring Boot 2.0
之前,我们都是直接继承
WebMvcConfifigurerAdapter
类,然后重写 addInterceptors 方法来实现拦截器的配置。但是在
Spring Boot 2.0
之后,该方法已经被废弃了 (当然,也可以继续用),取而代之的是 WebMvcConfifigurationSupport
方法,如下: 在该配置中重写 addInterceptors
方法,将我们上面自定义的拦截器添加进去,
addPathPatterns 方法是添加要拦截的请求,这里我们拦截所有的请求。这样就配置好拦截器了,接下来写一个 Controller 测试一下:
/**
* 自定义拦截器
*/
public class MyInterceptor implements HandlerInterceptor {
private static final Logger logger = LoggerFactory.getLogger(MyInterceptor.class);
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse
response, Object handler) throws Exception {
HandlerMethod handlerMethod = (HandlerMethod) handler;
Method method = handlerMethod.getMethod();
String methodName = method.getName();
logger.info("====拦截到了方法:{},在该方法执行之前执行====", methodName);
// 返回true才会继续执行,返回false则取消当前请求
return true;
}
@Override
public void postHandle(HttpServletRequest request, HttpServletResponse
response, Object handler, ModelAndView modelAndView) throws Exception {
logger.info("执行完方法之后进执行(Controller方法调用之后),但是此时还没进行视图渲染");
}
@Override
public void afterCompletion(HttpServletRequest request, HttpServletResponse
response, Object handler, Exception ex) throws Exception {
logger.info("整个请求都处理完咯,DispatcherServlet也渲染了对应的视图咯,此时我可 以做一些清理的工作了");
}
}
启动项目,postman请求,可以看到日志
3.
解决静态资源被拦截问题
上文中已经介绍了拦截器的定义和配置,但是这样是否就没问题了呢?其实不然,如果使用上面这种配 置的话,我们会发现一个缺陷,那就是静态资源被拦截了。可以在 resources/static/
目录下放置一个图 片资源或者 html
文件,然后启动项目直接访问,即可看到无法访问的现象。 也就是说,虽然 Spring Boot 2.0
废弃了
WebMvcConfifigurerAdapter
,但是WebMvcConfifigurationSupport 又会导致默认的静态资源被拦截,这就需要我们手动将静态资源放开。
如何放开呢?除了在
MyInterceptorConfifig
配置类中重写
addInterceptors
方法外,还需要再重写一个方法: addResourceHandlers
,将静态资源放开: 这样配置好之后,重启项目,静态资源也可以正常访问了。
/**
* 用来指定静态资源不被拦截,否则继承WebMvcConfigurationSupport这种方式会导致静态资源无法 直接访问
* @param registry
* */
@Override
protected void addResourceHandlers(ResourceHandlerRegistry registry) {
registry.addResourceHandler("/**").addResourceLocations("classpath:/static/");
super.addResourceHandlers(registry);
}
如果你是个善于学习或者研究的人,那肯定 不会止步于此,没错,上面这种方式的确能解决静态资源无法访问的问题,但是,还有更方便的方式来配置。我们不继承 WebMvcConfifigurationSupport 类,直接实现
WebMvcConfifigurer
接口,然后重写 addInterceptors 方法,将自定义的拦截器添加进去即可,如下:
@Override
public void addInterceptors(InterceptorRegistry registry) {
// 实现WebMvcConfigurer不会导致静态资源被拦截
registry.addInterceptor(new MyInterceptor()).addPathPatterns("/**");
}
这样就非常方便了,实现 WebMvcConfifigure 接口的话,不会拦截 Spring Boot 默认的静态资源。 这两种方式都可以,由于这两种方式的不 同,继承 WebMvcConfifigurationSupport 类的方式可以用在前后端分离的项目中,后台不需要访问静态资源(就不需要放开静态资源了);实现 WebMvcConfifigure 接口的方式可以用在非前后端分离的项目中,因为需要读取一些图片、css、js文件等等。