先看看拦截器都作些什么:html
1、日志记录:记录请求信息的日志,以便进行信息监控、信息统计、计算PV(PageView)等。java
2、权限检查:如登陆检测,进入处理器检测检测是否登陆,若是没有直接返回到登陆页面;web
3、性能监控:有时候系统在某段时间莫名其妙的慢,能够经过拦截器在进入处理器以前记录开始时间,在处理完后记录结束时间,从而获得该请求的处理时间(若是有反向代理,如apache能够自动记录);spring
4、通用行为:读取cookie获得用户信息并将用户对象放入请求,从而方便后续流程使用,还有如提取Locale、Theme信息等,只要是多个处理器都须要的便可使用拦截器实现。apache
5、OpenSessionInView:如Hibernate,在进入处理器打开Session,在完成后关闭Session。编程
本质也是AOP(面向切面编程),也就是说符合横切关注点的全部功能均可以放入拦截器实现。设计模式
Spring为咱们提供了:cookie
org.springframework.web.servlet.HandlerInterceptor接口,app
org.springframework.web.servlet.handler.HandlerInterceptorAdapter适配器,jsp
实现这个接口或继承此类,能够很是方便的实现本身的拦截器。
从源码中能够看到接口和类的关系:
从关系中能够看到AsyncHandlerInterceptor继承自HandlerInterceptor,而HandlerInterceptorAdapter又实现AsyncHandlerInterceptor,其实这是一种设计模式,叫适配器模式(不知道的能够点击查看适配器模式)。
如若实现HandlerInterceptor接口则三个方法都必须实现,若是继承HandlerInterceptorAdapter适配,则能够只实现咱们须要的回调方法。
从接口HandlerInterceptor中能够看到拦截器有三个回调方法,咱们来看看这3个都有什么做用:
preHandle:预处理回调方法,实现处理器的预处理(如登陆检查);方法返回值为boolean,返回true标示流程继续;返回false标示流程中断,好比登陆失 败,这样就不会再调用其它拦截器,咱们能够经过response来产生响应。
postHandle:后处理回调方法,实现处理器的后处理,此方法在生成视图以前执行;在这个方法中咱们能够修改modelAndView,经过modelAndView对模型数据进行处理或对视图进行处理。
afterCompletion:最后执行,可用于释放资源。能够根据ex是否为null判断是否发生了异常,进行日志记录。
参数中的Object handler是下一个拦截器。
自定义拦截器:
下面例子,用户登陆以后,经过拦截器对系统中的资源进行访问拦截,同时对于相关的资源不拦截
public class LoginInterceptor extends HandlerInterceptorAdapter { private static final String[] IGNORE_URI = {"/login.jsp", "/Login/","backui/","frontui/"}; @Override public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception { boolean flag = false; String url = request.getRequestURL().toString(); System.out.println(">>>: " + url); for (String s : IGNORE_URI) { if (url.contains(s)) { flag = true; break; } } if (!flag) { T_supplier_user user = LoginController.getLoginUser(request); if (user != null) flag = true; } return flag; //前面介绍返回false则会中断流程 } @Override public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception { super.postHandle(request, response, handler, modelAndView); } }
<interceptor> <mapping path="/**"/> <bean id="loginInterceptor" class="net.web.interceptor.LoginInterceptor"/> <interceptor>