目录
八、SpringMVC 中的拦截器
8.1 拦截器的作用
Spring MVC 的处理器拦截器类似于Servlet开发中的过滤器Filter,用于对处理器(Controller)进行预处理和后处理。
用户可以自己定义一些拦截器来实现特定的功能(权限控制、日志处理等)。
谈到拦截器,还要向大家提一个词——拦截器链(Interceptor Chain)。拦截器链就是将拦截器按一定的顺序联结成一条链。在访问被拦截的方法或字段时,拦截器链中的拦截器就会按其之前定义的顺序被调用。
过滤器和拦截器的区别:
区别1:
过滤器是servlet规范中的一部分,任何java web工程都可以使用。
拦截器是SpringMVC框架自己的,只有使用了SpringMVC框架的工程才能用。
区别2:
过滤器在url-pattern中配置了/*之后,可以对所有要访问的资源拦截。
拦截器它是只会拦截访问的控制器方法(只会拦截Controller)
核心控制器设置成/,表示拦截所有请求
核心控制器设置成*.do(*.action),表示拦截.do结尾的url请求
我们要想自定义拦截器, 要求必须实现:HandlerInterceptor接口。
8.2 自定义拦截器
8.2.1 数据准备
导入坐标
<!-- 版本锁定 -->
<properties>
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
<maven.compiler.source>1.8</maven.compiler.source>
<maven.compiler.target>1.8</maven.compiler.target>
<spring.version>5.0.2.RELEASE</spring.version>
</properties>
<dependencies>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-context</artifactId>
<version>${spring.version}</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-web</artifactId>
<version>${spring.version}</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-webmvc</artifactId>
<version>${spring.version}</version>
</dependency>
<dependency>
<groupId>javax.servlet</groupId>
<artifactId>servlet-api</artifactId>
<version>2.5</version>
<scope>provided</scope>
</dependency>
<dependency>
<groupId>javax.servlet.jsp</groupId>
<artifactId>jsp-api</artifactId>
<version>2.0</version>
<scope>provided</scope>
</dependency>
<dependency>
<groupId>jstl</groupId>
<artifactId>jstl</artifactId>
<version>1.2</version>
</dependency>
</dependencies>
web.xml
<!-- SpringMVC的核心控制器 -->
<servlet>
<servlet-name>dispatcherServlet</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<!-- 配置Servlet的初始化参数,读取springmvc的配置文件,创建spring容器 -->
<init-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:springmvc.xml</param-value>
</init-param>
<!-- 配置servlet启动时加载对象 -->
<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>dispatcherServlet</servlet-name>
<url-pattern>/</url-pattern>
</servlet-mapping>
<!--配置解决中文乱码过滤器-->
<filter>
<filter-name>characterEncodingFilter</filter-name>
<filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class>
<init-param>
<param-name>encoding</param-name>
<param-value>utf-8</param-value>
</init-param>
</filter>
<filter-mapping>
<filter-name>characterEncodingFilter</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>
springmvc.xml
<context:component-scan base-package="com.cpz"/>
<!--视图解析器-->
<bean id="viewResolver" class="org.springframework.web.servlet.view.InternalResourceViewResolver">
<property name="prefix" value="/WEB-INF/pages/"></property>
<property name="suffix" value=".jsp"></property>
</bean>
<mvc:annotation-driven></mvc:annotation-driven>
<!-- 设置静态资源不过滤 -->
<mvc:resources location="/js/" mapping="/js/**"/> <!-- javascript -->
Controller
@Controller
@RequestMapping(value = "/user")
public class UserController {
@RequestMapping(value = "/testInterceptor")
public String testInterceptor(){
System.out.println("执行了 testInterceptor 方法 ... ");
return "success";
}
}
index.jsp
<h1>拦截器</h1>
<a href="user/testInterceptor">自定义拦截器</a>
8.2.2 自定义拦截器
public class MyInterceptor1 implements HandlerInterceptor {
/**
* controller方法执行前,进行拦截的方法
* return true放行
* return false拦截
* 可以使用转发或者重定向直接跳转到指定的页面。
*/
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
System.out.println("1访问 Controller 方法之前...");
//request.getRequestDispatcher("/WEB-INF/pages/error.jsp").forward(request,response);
return true;
}
@Override
public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
System.out.println("1访问 Controller 方法之后 ...");
//request.getRequestDispatcher("/WEB-INF/pages/error.jsp").forward(request,response);
}
@Override
public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
System.out.println("1访问 success.jsp 之后 ... ");
// 在执行了 success.jsp 之后进行请求转发会报错
//request.getRequestDispatcher("/WEB-INF/pages/error.jsp").forward(request,response);
}
}
1:preHandle方法是controller方法执行前拦截的方法
可以使用request或者response跳转到指定的页面
return true放行,执行下一个拦截器,如果没有拦截器,执行controller中的方法。
return false不放行,不会执行controller中的方法。
2:postHandle是controller方法执行后执行的方法,在JSP视图执行前。
可以使用request或者response跳转到指定的页面
如果指定了跳转的页面,那么controller方法跳转的页面将不会显示。
3:afterCompletion方法是在JSP执行后执行(用的不多)
request或者response不能再跳转页面了,否则抛出异常。
8.2.3 配置拦截器类
<!--配置拦截器类-->
<mvc:interceptors>
<mvc:interceptor>
<!-- 哪些方法进行拦截 -->
<mvc:mapping path="/user/*"/>
<!--哪些方法不进行拦截-->
<!--<mvc:exclude-mapping path="/user/save"/>-->
<!-- 注册拦截器对象 -->
<bean class="com.cpz.interceptor.MyInterceptor1"></bean>
</mvc:interceptor>
</mvc:interceptors>
8.3 配置多个拦截器
8.3.1 再编写一个拦截器
public class MyInterceptor2 implements HandlerInterceptor {
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
System.out.println("2访问 Controller 方法之前...");
return true;
}
@Override
public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
System.out.println("2访问 Controller 方法之后 ...");
}
@Override
public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
System.out.println("2访问 success.jsp 之后 ... ");
}
}
8.3.2 配置拦截器
<mvc:interceptors>
<mvc:interceptor>
<!-- 哪些方法进行拦截 -->
<mvc:mapping path="/user/*"/>
<!--哪些方法不进行拦截-->
<!--<mvc:exclude-mapping path="/user/save"/>-->
<!-- 注册拦截器对象 -->
<bean class="com.cpz.interceptor.MyInterceptor1"></bean>
</mvc:interceptor>
<mvc:interceptor>
<!-- 哪些方法进行拦截 -->
<mvc:mapping path="/**"/>
<!--哪些方法不进行拦截-->
<!--<mvc:exclude-mapping path="/user/save"/>-->
<!-- 注册拦截器对象 -->
<bean class="com.cpz.interceptor.MyInterceptor2"></bean>
</mvc:interceptor>
</mvc:interceptors>