搜索 K
Appearance
博客正在加载中...
Appearance
Spring MVC 的处理器拦截器类似于 Servlet 开发中的过滤器 Filter,用于对处理器进行预处理和后处理。用户可以自己定义一些拦截器来实现特定的功能。
谈到拦截器,还要向大家提一个词——拦截器链(Interceptor Chain)。拦截器链就是将拦截器按一定的顺序联结成一条链。在访问被拦截的方法或字段时,拦截器链中的拦截器就会按其之前定义的顺序被调用。
说到这里,可能大家脑海中有了一个疑问,这不是我们之前学的过滤器吗?是的它和过滤器是有几分相似,但是也有区别,接下来我们就来说说他们的区别:
/* 之后,可以对所有要访问的资源拦截。拦截器也是 AOP 思想的具体应用。
我们要想自定义拦截器, 要求必须实现:HandlerInterceptor 接口。接下来我们演示如何用
我们可以删除之前的 Java 代码,并将配置文件中的异常处理的类删掉;
并且初始化 index.jsp 的内容为:
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>Title</title>
</head>
<body>
<h3>拦截器</h3>
<a href="user/testInterceptor">拦截器</a>
</body>
</html>success.jsp
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>Title</title>
</head>
<body>
<h3>执行成功</h3>
<% System.out.println("执行成功"); %>
</body>
</html>error.jsp
<%@ page contentType="text/html;charset=UTF-8" language="java" isELIgnored="false" %>
<html>
<head>
<title>Title</title>
</head>
<body>
<h3>错误页面</h3>
</body>
</html>
package com.peterjxl.controller;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
@Controller
@RequestMapping("/user")
public class UserController {
@RequestMapping("/testInterceptor")
public String testInterceptor() {
System.out.println("testInterceptor执行了...");
return "success";
}
}新建一个类,实现 HandlerInterceptor 接口;
package com.peterjxl.interceptor;
import org.springframework.web.servlet.HandlerInterceptor;
public class MyInterceptor implements HandlerInterceptor {} 我们实现后,可以看到 IDE 没有报错,也就是我们不用实现里面的方法,也没有报错;我们可以看到其源码中,是有方法定义的,但是已经实现了,这是 Java8 的一个增强:
package org.springframework.web.servlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.springframework.lang.Nullable;
public interface HandlerInterceptor {
default boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
return true;
}
default void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, @Nullable ModelAndView modelAndView) throws Exception {
}
default void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, @Nullable Exception ex) throws Exception {
}
} 该接口的 3 个方法的作用:
preHandle 是在控制器之前执行,返回值为 true 则表示放行,执行下一个拦截器,如果没有,则执行 controller。如果 return false,则不放行,并且我们可以用方法中的 req 和 resp 对象,跳转到某个错误页面上。例如可以判断是否登录。postHandle 是在控制器之后执行,可用于记录一些日志之类的afterCompletion 是在返回页面之后执行,可用于释放一些资源 我们实现一下其方法:@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
System.out.println("MyInterceptor执行了...");
return true;
} 在 springmvc.xml 中配置:
<!-- 配置拦截器 -->
<mvc:interceptors>
<!-- 拦截器第一个拦截器 -->
<mvc:interceptor>
<!-- 拦截的路径 -->
<mvc:mapping path="/user/*"/>
<!-- 排除不拦截的路径 -->
<bean class="com.peterjxl.interceptor.MyInterceptor"/>
</mvc:interceptor>
</mvc:interceptors>这样配置之后,就会拦截/user 路径的所有请求,而其他路径的请求不拦截。我们可以在 <mvc:interceptors> 中配置多个拦截器,每个 <mvc:interceptor> 标签中定义一个拦截器
除了配置拦截的路径之外,还可以配置不拦截的路径,这样除了这些不拦截的路径,其他路径都拦截:
<!-- 配置拦截器 -->
<mvc:interceptors>
<!-- 拦截器第一个拦截器 -->
<mvc:interceptor>
<!-- 拦截的路径 -->
<mvc:exclude-mapping path="/user/*"/> <!-- 排除拦截的路径 -->
<!-- 排除不拦截的路径 -->
<bean class="com.peterjxl.interceptor.MyInterceptor"/>
</mvc:interceptor>
</mvc:interceptors>
测试之前,我们可以先自己思考一下方法执行的顺序:
控制台输出:
MyInterceptor执行了...
testInterceptor执行了...
执行成功
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
System.out.println("MyInterceptor执行了...");
// return true;
request.getRequestDispatcher("/WEB-INF/pages/error.jsp").forward(request, response);
return false; // 拦截器拦截请求,不再执行controller方法
}
测试完后,我们改为放行,方便后续的测试。
postHandle postHandle 方法是在控制器方法执行后,success.jsp 执行前,执行的方法。 运行结果:
MyInterceptor执行了...
testInterceptor执行了...
MyInterceptor执行了...postHandle
执行成功其实在后处理方法中,跳转页面也是可以的:
@Override
public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
System.out.println("MyInterceptor执行了...postHandle");
request.getRequestDispatcher("/WEB-INF/pages/error.jsp").forward(request, response);
}
afterCompletion 方法 success.jsp 执行后,afterCompletion 方法会执行,我们实现一下:
@Override
public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
System.out.println("MyInterceptor执行了...afterCompletion");
}运行结果:
MyInterceptor执行了...
testInterceptor执行了...
MyInterceptor执行了...postHandle
执行成功
MyInterceptor执行了...afterCompletion 注意,此时页面已经响应完成,再设置跳转到其他页面是不会成功的。
本项目已将源码上传到 GitHub 和 Gitee 上。并且创建了分支 demo11,读者可以通过切换分支来查看本文的示例代码。