匹配URI拦截具体的HandleMapping ,[color=red]以下配置必须放在*-servlet.xml上[/color]
mvc:mapping path匹配规则详细记录:
3.2.1、path必须以/开头进行匹配,因为默认的是匹配URI。
3.2.2、path匹配目录使用**,如/**/代表匹配所有的一级目录。
3.2.3、path匹配文件使用*,如/**/*.htm代表匹配所有一级目录下的所有htm文件。
3.2.4、path可以匹配以指定文字开头或者结尾或者中间包含的URI,如:
/twov**/twov*.htm
/*wov*/two*.htm
/*wovv/*wovv.htm
SpringMVC中使用Interceptor拦截器 [url]http://haohaoxuexi.iteye.com/blog/1750680[/url]
自定义controller的方法参数注解 [url]http://karthikg.wordpress.com/2009/11/08/learn-to-customize-spring-mvc-controller-method-arguments/[/url]
[b][color=red]方法拦截器:MethodInterceptor[/color][/b] [url]http://panyongzheng.iteye.com/blog/1126708[/url]
SpringMVC 中的Interceptor 拦截器也是相当重要和相当有用的,它的主要作用是拦截用户的请求并进行相应的处理。比如通过它来进行权限验证,或者是来判断用户是否登陆,或者是像12306 那样子判断当前时间是否是购票时间。
[size=x-large][color=red] 一、定义Interceptor实现类[/color][/size]
SpringMVC 中的Interceptor 拦截请求是通过HandlerInterceptor 来实现的。在SpringMVC 中定义一个Interceptor 非常简单,主要有两种方式,
第一种方式是要定义的Interceptor类要实现了Spring 的[color=blue]HandlerInterceptor[/color] 接口,或者是这个类继承实现了HandlerInterceptor 接口的类,比如Spring 已经提供的实现了HandlerInterceptor 接口的抽象类HandlerInterceptorAdapter ;
第二种方式是实现Spring的[color=blue]WebRequestInterceptor[/color]接口,或者是继承实现了WebRequestInterceptor的类。
[size=large][color=blue] (一)实现HandlerInterceptor接口[/color][/size]
HandlerInterceptor 接口中定义了三个方法,我们就是通过这三个方法来对用户的请求进行拦截处理的。
[b] (1 )preHandle[/b] (HttpServletRequest request, HttpServletResponse response, Object handle) 方法,顾名思义,[color=blue]该方法将在请求处理之前进行调用[/color]。SpringMVC 中的Interceptor 是链式的调用的,在一个应用中或者说是在一个请求中可以同时存在多个Interceptor 。每个Interceptor 的调用会依据它的声明顺序依次执行,而且最先执行的都是Interceptor 中的preHandle 方法,所以可以在这个方法中进行一些前置初始化操作或者是对当前请求的一个预处理,也可以在这个方法中进行一些判断来决定请求是否要继续进行下去。该方法的返回值是布尔值Boolean 类型的,[color=blue]当它返回为false 时,表示请求结束,后续的Interceptor 和Controller 都不会再执行[/color];当返回值为true 时就会继续调用下一个Interceptor 的preHandle 方法,如果已经是最后一个Interceptor 的时候就会是调用当前请求的Controller 方法。
[b] (2 )postHandle[/b] (HttpServletRequest request, HttpServletResponse response, Object handle, ModelAndView modelAndView) 方法,由preHandle 方法的解释我们知道这个方法包括后面要说到的afterCompletion 方法都只能是在当前所属的Interceptor 的preHandle 方法的返回值为true 时才能被调用。postHandle 方法,顾名思义就是[color=blue]在当前请求进行处理之后,也就是Controller 方法调用之后执行,但是它会在DispatcherServlet 进行视图返回渲染之前被调用,所以我们可以在这个方法中对Controller 处理之后的ModelAndView 对象进行操作[/color]。postHandle 方法被调用的方向跟preHandle 是相反的,也就是说先声明的Interceptor 的postHandle 方法反而会后执行,这和Struts2 里面的Interceptor 的执行过程有点类型。Struts2 里面的Interceptor 的执行过程也是链式的,只是在Struts2 里面需要手动调用ActionInvocation 的invoke 方法来触发对下一个Interceptor 或者是Action 的调用,然后每一个Interceptor 中在invoke 方法调用之前的内容都是按照声明顺序执行的,而invoke 方法之后的内容就是反向的。
[b] (3 )afterCompletion[/b](HttpServletRequest request, HttpServletResponse response, Object handle, Exception ex) 方法,该方法也是需要当前对应的Interceptor 的preHandle 方法的返回值为true 时才会执行。顾名思义,[color=blue]该方法将在整个请求结束之后,也就是在DispatcherServlet 渲染了对应的视图之后执行[/color]。这个方法的主要作用是用于进行资源清理工作的。
下面是一个简单的代码说明:
[size=large][color=blue] (二)实现WebRequestInterceptor 接口[/color][/size]
WebRequestInterceptor 中也定义了三个方法,我们也是通过这三个方法来实现拦截的。这三个方法都传递了同一个参数WebRequest ,那么这个WebRequest 是什么呢?这个WebRequest 是Spring 定义的一个接口,它里面的方法定义都基本跟HttpServletRequest 一样,在WebRequestInterceptor 中对WebRequest 进行的所有操作都将同步到HttpServletRequest 中,然后在当前请求中一直传递。
[b] (1 )preHandle[/b](WebRequest request) 方法。该方法将在请求处理之前进行调用,也就是说会在Controller 方法调用之前被调用。这个方法跟HandlerInterceptor 中的preHandle 是不同的,主要区别在于[color=blue]该方法的返回值是void ,也就是没有返回值,所以我们一般主要用它来进行资源的准备工作[/color],比如我们在使用Hibernate 的时候可以在这个方法中准备一个Hibernate 的Session 对象,然后利用WebRequest 的setAttribute(name, value, scope) 把它放到WebRequest 的属性中。这里可以说说这个setAttribute 方法的第三个参数scope ,该参数是一个Integer 类型的。在WebRequest 的父层接口RequestAttributes 中对它定义了三个常量:
SCOPE_REQUEST :它的值是0 ,代表只有在request 中可以访问。
SCOPE_SESSION :它的值是1 ,如果环境允许的话它代表的是一个局部的隔离的session,否则就代表普通的session,并且在该session范围内可以访问。
SCOPE_GLOBAL_SESSION :它的值是2 ,如果环境允许的话,它代表的是一个全局共享的session,否则就代表普通的session,并且在该session 范围内可以访问。
[b] (2 )postHandle[/b](WebRequest request, ModelMap model) 方法。[color=blue]该方法将在请求处理之后,也就是在Controller 方法调用之后被调用,但是会在视图返回被渲染之前被调用,所以可以在这个方法里面通过改变数据模型ModelMap 来改变数据的展示[/color]。该方法有两个参数,WebRequest 对象是用于传递整个请求数据的,比如在preHandle 中准备的数据都可以通过WebRequest 来传递和访问;ModelMap 就是Controller 处理之后返回的Model 对象,我们可以通过改变它的属性来改变返回的Model 模型。
[b] (3 )afterCompletion[/b](WebRequest request, Exception ex) 方法。[color=blue]该方法会在整个请求处理完成,也就是在视图返回并被渲染之后执行[/color]。所以在该方法中可以进行资源的释放操作。而WebRequest 参数就可以把我们在preHandle 中准备的资源传递到这里进行释放。Exception 参数表示的是当前请求的异常对象,如果在Controller 中抛出的异常已经被Spring 的异常处理器给处理了的话,那么这个异常对象就是是null 。
下面是一个简单的代码说明:
[size=x-large][color=red] 二、把定义的拦截器类加到SpringMVC的拦截体系中[/color][/size]
[size=large][color=blue] 1.在SpringMVC的配置文件中加上支持MVC的schema[/color][/size]
下面是我的声明示例:
这样在SpringMVC的配置文件中就可以使用mvc标签了,mvc标签中有一个mvc:interceptors是用于声明SpringMVC的拦截器的。
[size=large][color=blue] (二)使用mvc:interceptors标签来声明需要加入到SpringMVC拦截器链中的拦截器[/color][/size]
由上面的示例可以看出可以利用mvc:interceptors标签声明一系列的拦截器,然后它们就可以形成一个拦截器链,拦截器的执行顺序是按声明的先后顺序执行的,先声明的拦截器中的preHandle方法会先执行,然而它的postHandle方法和afterCompletion方法却会后执行。
[b] 在mvc:interceptors标签下声明interceptor主要有两种方式:[/b]
(1)直接定义一个Interceptor实现类的bean对象。使用这种方式声明的Interceptor拦截器将会对所有的请求进行拦截。
(2)使用mvc:interceptor标签进行声明。使用这种方式进行声明的Interceptor可以通过mvc:mapping子标签来定义需要进行拦截的请求路径。
经过上述两步之后,定义的拦截器就会发生作用对特定的请求进行拦截了。
[size=x-large][color=red]我的一个例子源码:[/color][/size]
servlet.xml
定义注解:
处理注解的类:
两个全局拦截器类:
一个特殊场合的拦截器:
controller:
<mvc:interceptors>
<mvc:interceptor>
<mvc:mapping path="/test/interceptor.htm"/>
<bean class="com.twovv.intercepter.MyImplementInterceptor"/>
</mvc:interceptor>
</mvc:interceptors>
mvc:mapping path匹配规则详细记录:
3.2.1、path必须以/开头进行匹配,因为默认的是匹配URI。
3.2.2、path匹配目录使用**,如/**/代表匹配所有的一级目录。
3.2.3、path匹配文件使用*,如/**/*.htm代表匹配所有一级目录下的所有htm文件。
3.2.4、path可以匹配以指定文字开头或者结尾或者中间包含的URI,如:
/twov**/twov*.htm
/*wov*/two*.htm
/*wovv/*wovv.htm
SpringMVC中使用Interceptor拦截器 [url]http://haohaoxuexi.iteye.com/blog/1750680[/url]
自定义controller的方法参数注解 [url]http://karthikg.wordpress.com/2009/11/08/learn-to-customize-spring-mvc-controller-method-arguments/[/url]
[b][color=red]方法拦截器:MethodInterceptor[/color][/b] [url]http://panyongzheng.iteye.com/blog/1126708[/url]
import org.aopalliance.intercept.MethodInterceptor;
import org.aopalliance.intercept.MethodInvocation;
SpringMVC 中的Interceptor 拦截器也是相当重要和相当有用的,它的主要作用是拦截用户的请求并进行相应的处理。比如通过它来进行权限验证,或者是来判断用户是否登陆,或者是像12306 那样子判断当前时间是否是购票时间。
[size=x-large][color=red] 一、定义Interceptor实现类[/color][/size]
SpringMVC 中的Interceptor 拦截请求是通过HandlerInterceptor 来实现的。在SpringMVC 中定义一个Interceptor 非常简单,主要有两种方式,
第一种方式是要定义的Interceptor类要实现了Spring 的[color=blue]HandlerInterceptor[/color] 接口,或者是这个类继承实现了HandlerInterceptor 接口的类,比如Spring 已经提供的实现了HandlerInterceptor 接口的抽象类HandlerInterceptorAdapter ;
第二种方式是实现Spring的[color=blue]WebRequestInterceptor[/color]接口,或者是继承实现了WebRequestInterceptor的类。
[size=large][color=blue] (一)实现HandlerInterceptor接口[/color][/size]
HandlerInterceptor 接口中定义了三个方法,我们就是通过这三个方法来对用户的请求进行拦截处理的。
[b] (1 )preHandle[/b] (HttpServletRequest request, HttpServletResponse response, Object handle) 方法,顾名思义,[color=blue]该方法将在请求处理之前进行调用[/color]。SpringMVC 中的Interceptor 是链式的调用的,在一个应用中或者说是在一个请求中可以同时存在多个Interceptor 。每个Interceptor 的调用会依据它的声明顺序依次执行,而且最先执行的都是Interceptor 中的preHandle 方法,所以可以在这个方法中进行一些前置初始化操作或者是对当前请求的一个预处理,也可以在这个方法中进行一些判断来决定请求是否要继续进行下去。该方法的返回值是布尔值Boolean 类型的,[color=blue]当它返回为false 时,表示请求结束,后续的Interceptor 和Controller 都不会再执行[/color];当返回值为true 时就会继续调用下一个Interceptor 的preHandle 方法,如果已经是最后一个Interceptor 的时候就会是调用当前请求的Controller 方法。
[b] (2 )postHandle[/b] (HttpServletRequest request, HttpServletResponse response, Object handle, ModelAndView modelAndView) 方法,由preHandle 方法的解释我们知道这个方法包括后面要说到的afterCompletion 方法都只能是在当前所属的Interceptor 的preHandle 方法的返回值为true 时才能被调用。postHandle 方法,顾名思义就是[color=blue]在当前请求进行处理之后,也就是Controller 方法调用之后执行,但是它会在DispatcherServlet 进行视图返回渲染之前被调用,所以我们可以在这个方法中对Controller 处理之后的ModelAndView 对象进行操作[/color]。postHandle 方法被调用的方向跟preHandle 是相反的,也就是说先声明的Interceptor 的postHandle 方法反而会后执行,这和Struts2 里面的Interceptor 的执行过程有点类型。Struts2 里面的Interceptor 的执行过程也是链式的,只是在Struts2 里面需要手动调用ActionInvocation 的invoke 方法来触发对下一个Interceptor 或者是Action 的调用,然后每一个Interceptor 中在invoke 方法调用之前的内容都是按照声明顺序执行的,而invoke 方法之后的内容就是反向的。
[b] (3 )afterCompletion[/b](HttpServletRequest request, HttpServletResponse response, Object handle, Exception ex) 方法,该方法也是需要当前对应的Interceptor 的preHandle 方法的返回值为true 时才会执行。顾名思义,[color=blue]该方法将在整个请求结束之后,也就是在DispatcherServlet 渲染了对应的视图之后执行[/color]。这个方法的主要作用是用于进行资源清理工作的。
下面是一个简单的代码说明:
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.springframework.web.servlet.HandlerInterceptor;
import org.springframework.web.servlet.ModelAndView;
public class SpringMVCInterceptor implements HandlerInterceptor {
/**
* preHandle方法是进行处理器拦截用的,顾名思义,该方法将在Controller处理之前进行调用,SpringMVC中的Interceptor拦截器是链式的,可以同时存在
* 多个Interceptor,然后SpringMVC会根据声明的前后顺序一个接一个的执行,而且所有的Interceptor中的preHandle方法都会在
* Controller方法调用之前调用。SpringMVC的这种Interceptor链式结构也是可以进行中断的,这种中断方式是令preHandle的返
* 回值为false,当preHandle的返回值为false的时候整个请求就结束了。
*/
@Override
public boolean preHandle(HttpServletRequest request,
HttpServletResponse response, Object handler) throws Exception {
// TODO Auto-generated method stub
return false;
}
/**
* 这个方法只会在当前这个Interceptor的preHandle方法返回值为true的时候才会执行。postHandle是进行处理器拦截用的,它的执行时间是在处理器进行处理之
* 后,也就是在Controller的方法调用之后执行,但是它会在DispatcherServlet进行视图的渲染之前执行,也就是说在这个方法中你可以对ModelAndView进行操
* 作。这个方法的链式结构跟正常访问的方向是相反的,也就是说先声明的Interceptor拦截器该方法反而会后调用,这跟Struts2里面的拦截器的执行过程有点像,
* 只是Struts2里面的intercept方法中要手动的调用ActionInvocation的invoke方法,Struts2中调用ActionInvocation的invoke方法就是调用下一个Interceptor
* 或者是调用action,然后要在Interceptor之前调用的内容都写在调用invoke之前,要在Interceptor之后调用的内容都写在调用invoke方法之后。
*/
@Override
public void postHandle(HttpServletRequest request,
HttpServletResponse response, Object handler,
ModelAndView modelAndView) throws Exception {
// TODO Auto-generated method stub
}
/**
* 该方法也是需要当前对应的Interceptor的preHandle方法的返回值为true时才会执行。该方法将在整个请求完成之后,也就是DispatcherServlet渲染了视图执行,
* 这个方法的主要作用是用于清理资源的,当然这个方法也只能在当前这个Interceptor的preHandle方法的返回值为true时才会执行。
*/
@Override
public void afterCompletion(HttpServletRequest request,
HttpServletResponse response, Object handler, Exception ex)
throws Exception {
// TODO Auto-generated method stub
}
}
[size=large][color=blue] (二)实现WebRequestInterceptor 接口[/color][/size]
WebRequestInterceptor 中也定义了三个方法,我们也是通过这三个方法来实现拦截的。这三个方法都传递了同一个参数WebRequest ,那么这个WebRequest 是什么呢?这个WebRequest 是Spring 定义的一个接口,它里面的方法定义都基本跟HttpServletRequest 一样,在WebRequestInterceptor 中对WebRequest 进行的所有操作都将同步到HttpServletRequest 中,然后在当前请求中一直传递。
[b] (1 )preHandle[/b](WebRequest request) 方法。该方法将在请求处理之前进行调用,也就是说会在Controller 方法调用之前被调用。这个方法跟HandlerInterceptor 中的preHandle 是不同的,主要区别在于[color=blue]该方法的返回值是void ,也就是没有返回值,所以我们一般主要用它来进行资源的准备工作[/color],比如我们在使用Hibernate 的时候可以在这个方法中准备一个Hibernate 的Session 对象,然后利用WebRequest 的setAttribute(name, value, scope) 把它放到WebRequest 的属性中。这里可以说说这个setAttribute 方法的第三个参数scope ,该参数是一个Integer 类型的。在WebRequest 的父层接口RequestAttributes 中对它定义了三个常量:
SCOPE_REQUEST :它的值是0 ,代表只有在request 中可以访问。
SCOPE_SESSION :它的值是1 ,如果环境允许的话它代表的是一个局部的隔离的session,否则就代表普通的session,并且在该session范围内可以访问。
SCOPE_GLOBAL_SESSION :它的值是2 ,如果环境允许的话,它代表的是一个全局共享的session,否则就代表普通的session,并且在该session 范围内可以访问。
[b] (2 )postHandle[/b](WebRequest request, ModelMap model) 方法。[color=blue]该方法将在请求处理之后,也就是在Controller 方法调用之后被调用,但是会在视图返回被渲染之前被调用,所以可以在这个方法里面通过改变数据模型ModelMap 来改变数据的展示[/color]。该方法有两个参数,WebRequest 对象是用于传递整个请求数据的,比如在preHandle 中准备的数据都可以通过WebRequest 来传递和访问;ModelMap 就是Controller 处理之后返回的Model 对象,我们可以通过改变它的属性来改变返回的Model 模型。
[b] (3 )afterCompletion[/b](WebRequest request, Exception ex) 方法。[color=blue]该方法会在整个请求处理完成,也就是在视图返回并被渲染之后执行[/color]。所以在该方法中可以进行资源的释放操作。而WebRequest 参数就可以把我们在preHandle 中准备的资源传递到这里进行释放。Exception 参数表示的是当前请求的异常对象,如果在Controller 中抛出的异常已经被Spring 的异常处理器给处理了的话,那么这个异常对象就是是null 。
下面是一个简单的代码说明:
import org.springframework.ui.ModelMap;
import org.springframework.web.context.request.WebRequest;
import org.springframework.web.context.request.WebRequestInterceptor;
public class AllInterceptor implements WebRequestInterceptor {
/**
* 在请求处理之前执行,该方法主要是用于准备资源数据的,然后可以把它们当做请求属性放到WebRequest中
*/
@Override
public void preHandle(WebRequest request) throws Exception {
// TODO Auto-generated method stub
System.out.println("AllInterceptor...............................");
request.setAttribute("request", "request", WebRequest.SCOPE_REQUEST);//这个是放到request范围内的,所以只能在当前请求中的request中获取到
request.setAttribute("session", "session", WebRequest.SCOPE_SESSION);//这个是放到session范围内的,如果环境允许的话它只能在局部的隔离的会话中访问,否则就是在普通的当前会话中可以访问
request.setAttribute("globalSession", "globalSession", WebRequest.SCOPE_GLOBAL_SESSION);//如果环境允许的话,它能在全局共享的会话中访问,否则就是在普通的当前会话中访问
}
/**
* 该方法将在Controller执行之后,返回视图之前执行,ModelMap表示请求Controller处理之后返回的Model对象,所以可以在
* 这个方法中修改ModelMap的属性,从而达到改变返回的模型的效果。
*/
@Override
public void postHandle(WebRequest request, ModelMap map) throws Exception {
// TODO Auto-generated method stub
for (String key:map.keySet())
System.out.println(key + "-------------------------");;
map.put("name3", "value3");
map.put("name1", "name1");
}
/**
* 该方法将在整个请求完成之后,也就是说在视图渲染之后进行调用,主要用于进行一些资源的释放
*/
@Override
public void afterCompletion(WebRequest request, Exception exception)
throws Exception {
// TODO Auto-generated method stub
System.out.println(exception + "-=-=--=--=-=-=-=-=-=-=-=-==-=--=-=-=-=");
}
}
[size=x-large][color=red] 二、把定义的拦截器类加到SpringMVC的拦截体系中[/color][/size]
[size=large][color=blue] 1.在SpringMVC的配置文件中加上支持MVC的schema[/color][/size]
xmlns:mvc="http://www.springframework.org/schema/mvc"
xsi:schemaLocation=" http://www.springframework.org/schema/mvc
http://www.springframework.org/schema/mvc/spring-mvc-3.0.xsd"
下面是我的声明示例:
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:context="http://www.springframework.org/schema/context"
xmlns:mvc="http://www.springframework.org/schema/mvc"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-3.0.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context-3.0.xsd
http://www.springframework.org/schema/mvc
http://www.springframework.org/schema/mvc/spring-mvc-3.0.xsd">
这样在SpringMVC的配置文件中就可以使用mvc标签了,mvc标签中有一个mvc:interceptors是用于声明SpringMVC的拦截器的。
[size=large][color=blue] (二)使用mvc:interceptors标签来声明需要加入到SpringMVC拦截器链中的拦截器[/color][/size]
<mvc:interceptors>
<!-- 使用bean定义一个Interceptor,直接定义在mvc:interceptors根下面的Interceptor将拦截所有的请求 -->
<bean class="com.host.app.web.interceptor.AllInterceptor"/>
<mvc:interceptor>
<mvc:mapping path="/test/number.do"/>
<!-- 定义在mvc:interceptor下面的表示是对特定的请求才进行拦截的 -->
<bean class="com.host.app.web.interceptor.LoginInterceptor"/>
</mvc:interceptor>
</mvc:interceptors>
由上面的示例可以看出可以利用mvc:interceptors标签声明一系列的拦截器,然后它们就可以形成一个拦截器链,拦截器的执行顺序是按声明的先后顺序执行的,先声明的拦截器中的preHandle方法会先执行,然而它的postHandle方法和afterCompletion方法却会后执行。
[b] 在mvc:interceptors标签下声明interceptor主要有两种方式:[/b]
(1)直接定义一个Interceptor实现类的bean对象。使用这种方式声明的Interceptor拦截器将会对所有的请求进行拦截。
(2)使用mvc:interceptor标签进行声明。使用这种方式进行声明的Interceptor可以通过mvc:mapping子标签来定义需要进行拦截的请求路径。
经过上述两步之后,定义的拦截器就会发生作用对特定的请求进行拦截了。
[size=x-large][color=red]我的一个例子源码:[/color][/size]
servlet.xml
<!-- 拦截器 -->
<mvc:annotation-driven>
<mvc:argument-resolvers>
<bean class="com.interceptor.SessionParamArgumentResolver" />
</mvc:argument-resolvers>
</mvc:annotation-driven>
<mvc:interceptors>
<!-- 多个拦截器,顺序执行 -->
<bean class="com.interceptor.SpringMVCHandlerInterceptor"/>
<bean class="com.interceptor.SpringMVCWebRequestInterceptor"/>
<mvc:interceptor>
<mvc:mapping path="/itc/*"/>
<!-- 如果不配置或/*,将拦截所有的Controller -->
<bean class="com.controller.CommonInterceptor"></bean>
</mvc:interceptor>
</mvc:interceptors>
定义注解:
package com.interceptor;
import java.lang.annotation.*;
/**
* //一个注解的定义
*/
@Target(ElementType.PARAMETER)
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface MySessionParamAnn {
/**
* //设定value属性的方法
*/
String value() default "";
/**
* //设定required属性的方法
*/
boolean required() default true;
/**
* //设定defaultValue属性的方法
*/
String defaultValue() default "";
}
处理注解的类:
package com.interceptor;
import org.springframework.core.MethodParameter;
import org.springframework.web.HttpSessionRequiredException;
import org.springframework.web.bind.support.WebArgumentResolver;
import org.springframework.web.context.request.NativeWebRequest;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpSession;
import java.lang.annotation.Annotation;
/**
* //对注解赋值时候的解析和判断处理
*/
public class SessionParamArgumentResolver implements WebArgumentResolver {
@Override
public Object resolveArgument(MethodParameter param, NativeWebRequest request) throws Exception {
System.out.println("------------------->SessionParamArgumentResolver[resolveArgument]");
//查找在方法参数中指定的注释
Annotation[] paramAnns = param.getParameterAnnotations();
Class paramType = param.getParameterType();
for (Annotation paramAnn : paramAnns) {
//如果检查方法参数标注的类型是@ SessionParam的
if (MySessionParamAnn.class.isInstance(paramAnn)) {
MySessionParamAnn mySessionParamAnn = (MySessionParamAnn) paramAnn;
String paramName = mySessionParamAnn.value();
boolean required = mySessionParamAnn.required();
String defaultValue = mySessionParamAnn.defaultValue();
HttpServletRequest httprequest = (HttpServletRequest) request
.getNativeRequest();
HttpSession session = httprequest.getSession(false);
Object result = null;
if (session != null) {
result = session.getAttribute(paramName);
}
if (result == null)
result = defaultValue;
if (result == null && required && session == null) {
//抛出一个异常,如果是required,并且没有没有注入有效值
raiseSessionRequiredException(paramName, paramType);
}
if (result == null && required) {
//如果未在HTTP会话中发现的参数抛出一个异常
raiseMissingParameterException(paramName, paramType);
}
System.out.println("------------------->SessionParamArgumentResolver[resolveArgument] # Annotation Return");
return result;
}
}
System.out.println("------------------->SessionParamArgumentResolver[resolveArgument] # UNRESOLVED");
return WebArgumentResolver.UNRESOLVED;
}
protected void raiseMissingParameterException(String paramName, Class paramType) throws Exception {
throw new IllegalStateException("Missing parameter '" + paramName
+ "' of type [" + paramType.getName() + "]");
}
protected void raiseSessionRequiredException(String paramName, Class paramType) throws Exception {
throw new HttpSessionRequiredException(
"No HttpSession found for resolving parameter '" + paramName
+ "' of type [" + paramType.getName() + "]");
}
}
两个全局拦截器类:
package com.interceptor;
import org.springframework.web.servlet.HandlerInterceptor;
import org.springframework.web.servlet.ModelAndView;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
public class SpringMVCHandlerInterceptor implements HandlerInterceptor {
@Override
public boolean preHandle(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse, Object o) throws Exception {
System.out.println("## SpringMVCHandlerInterceptor[preHandle]");
return true;
}
@Override
public void postHandle(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse, Object o, ModelAndView modelAndView) throws Exception {
System.out.println("## SpringMVCHandlerInterceptor[postHandle]");
}
@Override
public void afterCompletion(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse, Object o, Exception e) throws Exception {
System.out.println("## SpringMVCHandlerInterceptor[afterCompletion]");
}
}
package com.interceptor;
import org.springframework.ui.ModelMap;
import org.springframework.web.context.request.WebRequest;
import org.springframework.web.context.request.WebRequestInterceptor;
/**
* Created by Administrator on 2014/6/25.
*/
public class SpringMVCWebRequestInterceptor implements WebRequestInterceptor {
@Override
public void preHandle(WebRequest webRequest) throws Exception {
System.out.println("## SpringMVCWebRequestInterceptor[preHandle]");
}
@Override
public void postHandle(WebRequest webRequest, ModelMap modelMap) throws Exception {
System.out.println("## SpringMVCWebRequestInterceptor[postHandle]");
}
@Override
public void afterCompletion(WebRequest webRequest, Exception e) throws Exception {
System.out.println("## SpringMVCWebRequestInterceptor[afterCompletion]");
}
}
一个特殊场合的拦截器:
package com.controller;
/**
* Created by Administrator on 14-6-15.
*/
import org.apache.log4j.Logger;
import org.springframework.web.servlet.HandlerInterceptor;
import org.springframework.web.servlet.ModelAndView;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
public class CommonInterceptor implements HandlerInterceptor {
private Logger log = Logger.getLogger(CommonInterceptor.class);
public CommonInterceptor() {
// TODO Auto-generated constructor stub
}
private String mappingURL;//利用正则映射到需要拦截的路径
public void setMappingURL(String mappingURL) {
this.mappingURL = mappingURL;
}
/**
* 在业务处理器处理请求之前被调用
* 如果返回false
* 从当前的拦截器往回执行所有拦截器的afterCompletion(),再退出拦截器链
*
* 如果返回true
* 执行下一个拦截器,直到所有的拦截器都执行完毕
* 再执行被拦截的Controller
* 然后进入拦截器链,
* 从最后一个拦截器往回执行所有的postHandle()
* 接着再从最后一个拦截器往回执行所有的afterCompletion()
*/
@Override
public boolean preHandle(HttpServletRequest request,
HttpServletResponse response, Object handler) throws Exception {
// TODO Auto-generated method stub
log.info("==============执行顺序: 1、preHandle================");
String url=request.getRequestURL().toString();
if(mappingURL==null || url.matches(mappingURL)){
request.getRequestDispatcher("/msg.jsp").forward(request, response);
return false;
}
return true;
}
//在业务处理器处理请求执行完成后,生成视图之前执行的动作
@Override
public void postHandle(HttpServletRequest request,
HttpServletResponse response, Object handler,
ModelAndView modelAndView) throws Exception {
// TODO Auto-generated method stub
log.info("==============执行顺序: 2、postHandle================");
}
/**
* 在DispatcherServlet完全处理完请求后被调用
*
* 当有拦截器抛出异常时,会从当前拦截器往回执行所有的拦截器的afterCompletion()
*/
@Override
public void afterCompletion(HttpServletRequest request,
HttpServletResponse response, Object handler, Exception ex)
throws Exception {
// TODO Auto-generated method stub
log.info("==============执行顺序: 3、afterCompletion================");
}
}
controller:
package com.controller;
import com.interceptor.MySessionParamAnn;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.servlet.ModelAndView;
/**
* Created by Administrator on 14-6-15.
*/
@Controller
@RequestMapping("/webarg")
public class WebArgumentResolverController {
@RequestMapping("/test")
public ModelAndView test(@MySessionParamAnn(value="user") String user) {
System.out.println("自定义controller参数的注解:user="+user);
return new ModelAndView("index");
}
}