您的位置:首页 > 编程语言 > Java开发

springMVC拦截器

2015-06-20 12:51 411 查看

springMVC拦截器

从业2年多,第一次这么认真严肃的写技术博客。原谅我,来晚了。
一个程序员,如果不留下点什么,那这一生真是白活了,so,I'm here!


昨天刚刚系统学习了一下springMVC的拦截器,我的第六感觉得,servlet的过滤器是它与struts2的拦截器共同的祖先。

他们都是在一个动作执行之前、之后做点什么东西。

凭这点特性,可以利用拦截器做这些事情:

日志记录:用户、管理员,任何你想监控的人,在你的系统中何时、何地、干了些什么事,调用了哪个类那个方法,传递了什么参数,方法执行之前是什么状态,干完又是什么状态,这些东西都可以记录下来,作为日志,或用日志组件输出到控制台,本地磁盘文件系统,或者记录到数据库

权限检查:如用户进入一个页面或者请求一个url时,先对这个用户有没有权限做一下检测,如果有,放行,如果没有,拒之门外

性能监控:在这个系统中,做完一系列动作,花费了多长时间,这个时间怎么算?从进入这个拦截器开始,经过层层拦截器,到达web层Controller,然后Controller调用Service里的一系列方法,请求完成在经过层层拦截器,最后回到这个性能监测拦截器为止。(和AOP实现性能监测不同的是AOP方式计算时间恰好是service方法或者controller方法执行占用的时间,而没有一些各种拦截器的时间消耗。可以说,aop方式的性能检测更专注,而拦截器方式的性能监测方式眼光更远,它从整个系统层面来看待问题的。至于选哪种,根据自己需要咯,这些都是个人猜测



通用行为

OpenSessionInView:如Hibernate,在进入处理器打开Session,在完成后关闭Session。可以说这一条是上一条的一个特例

拦截器的2中实现方式:
一:实现HandlerInterceptor接口
你的类实现的这个接口,自然要实现这个接口下的方法
//处理器的预处理,返回值:true 表示继续流程false 表示流程中断不会继续调用其他的拦截器或处理器,此时我们需要通过response来产生响应
boolean preHandle(HttpServletRequest request, HttpServletResponse response,Object handler)throws Exception;
 //生成视图之前执行.此时我们可以通过modelAndView(模型和视图对象)对模型数据进行处理或对视图进行处理,modelAndView 也可能为null。


void postHandle(HttpServletRequest request, HttpServletResponse response,Object handler, ModelAndView modelAndView)throws Exception; 
//视图渲染完毕时回调如性能监控中我们可以在此记录结束时间并输出消耗时间还可以进行一些资源清理类似于try-catch-finally 中的finally的作用,表示最后总要执行的动作
void afterCompletion(HttpServletRequest request, HttpServletResponse response,Object handler, Exception ex) throws Exception;
}
<span style="font-size:18px;">二:继承HandlerInterceptorAdapter类

有时候我们可能只需要实现三个回调方法中的某一个,如果实现HandlerInterceptor 接口的话,三个方法必须实现,不管你需不需要,此时spring 提供了一个HandlerInterceptorAdapter适配器(一种适配器设计模式的实现),允许我们只实现需要的回调方法。
比如在一个授权拦截器中,用户请求的url之前,在这个拦截器的
preHandle(HttpServletRequest request, HttpServletResponse response, Object handler)

方法内判断当前是否有登录用户、当前的用户是否是否有请求这个url的权限等等判断即可
而postHandle,afterCompletion方法根本无需实现。
其实HandlerInterceptorAdapter对这些方法做了空实现,只是需求只需重写preHandle方法就行。

一个性能检测拉涅其的简单实现
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.web.servlet.handler.HandlerInterceptorAdapter;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

/**
* 请求性能检测器
*/
public class MonitorInterceptor extends HandlerInterceptorAdapter {
</span>
<span style="font-size:14px;">	Logger logger= LoggerFactory.getLogger(ActionLogInterceptor.class);</span>
<span style="font-size:14px;">    	//解决多线程变量共享的问题
private ThreadLocal<Long> threadLocal=new ThreadLocal<Long>();

@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
Long start=System.currentTimeMillis();
threadLocal.set(start);
return true;
}

@Override
public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
Long times=System.currentTimeMillis()-threadLocal.get();
logger.info("请求uri:{}花费:{}毫秒",request.getRequestURI(),times);
super.afterCompletion(request, response, handler, ex);
}
}


配置拦截器

springmvc-servlet.xml
<mvc:interceptors>
<!--性能检测-->
<bean class="com.interceptor.MonitorInterceptor"></bean>
</mvc:interceptors>


这样springMVC框架截获的所有url请求先经过这个拦截器

如果直线这个拦截器拦截部分url可以这样配置:
<mvc:interceptors>
<mvc:interceptor>
<mvc:mapping path="/test/**"/>

<bean class="com.interceptor.MonitorInterceptor"></bean>
</mvc:interceptor>
</mvc:interceptors>\


这样这个拦截器只拦截 url为 “/test/" 开头的url
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: