本文最后更新于 2023-09-22,文章内容可能已经过时。

SpringMVC《源码剖析》6. 拦截器和异常处理器

一、拦截器

1、拦截器的配置

/ :拦截所有请求 包括静态资源和动态请求 但是不拦截jsp
/*:拦截所有请求所有的文件夹,但是不包括子文件夹 包括静态资源和动态请求 也拦截jsp

/** :的意思是所有文件夹及里面的子文件夹

SpringMVC中的拦截器用于拦截控制器方法的执行

SpringMVC中的拦截器需要实现HandlerInterceptor

SpringMVC的拦截器必须在SpringMVC的配置文件中进行配置:

这里老师带看了源码,可以继承已经过时的HandlerInterceptorAdapter类,或者实现(推荐)HandlerInterceptor接口.重写它的三个方法,这里是默认方法所以不会强制重写(接口的默认方法有方法体)视频 P79看一下很简短的带看

public class FirstInterceptor implements HandlerInterceptor {
    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
        System.out.println("FirstInterceptor-->preHandle");
        return HandlerInterceptor.super.preHandle(request, response, handler);
    }

    @Override
    public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
        System.out.println("FirstInterceptor-->postHandle");
        HandlerInterceptor.super.postHandle(request, response, handler, modelAndView);
    }

    @Override
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
        System.out.println("FirstInterceptor-->afterCompletion");
        HandlerInterceptor.super.afterCompletion(request, response, handler, ex);
    }
}
@Controller
public class TestController {

    @RequestMapping("/**/testInterceptor")
    public String testInterceptor(){
        return "succ";
    }
}

拦截器配置:基础配置方法
这种时候如果Controller是上面的多层目录,也会拦截,甚至不存在的404资源也能拦截。但是浏览器端是404,方法其实还是会执行。

<mvc:interceptors>
        <bean class="com.fanxy.mvc.interceptor.FirstInterceptor"></bean>
    </mvc:interceptors>

<!-- 使用如下方法需要这个bean放在ioc容器中,要么通过注解然后开启组件扫描,要么通过spring的xml配置bean -->
<mvc:interceptors>
        <ref bean="firstInterceptor"></ref>
</mvc:interceptors>

<!-- 以上两种配置方式都是对DispatcherServlet所处理的所有的请求进行拦截 -->

筛选拦截请求
此时能匹配任意多层路径除了/testRequestEntity,但是如果是配置的/*,多层路径就不能拦截了

<bean class="com.fanxy.interceptor.FirstInterceptor"></bean>
<mvc:interceptors>

<mvc:interceptor>
    <mvc:mapping path="/**"/>
    <mvc:exclude-mapping path="/testRequestEntity"/>
    <ref bean="firstInterceptor"></ref>
</mvc:interceptor>

</mvc:interceptors>
<!-- 
	以上配置方式可以通过ref或bean标签设置拦截器,通过mvc:mapping设置需要拦截的请求,通过mvc:exclude-mapping设置需要排除的请求,即不需要拦截的请求
-->

2、拦截器的三个抽象方法

SpringMVC中的拦截器有三个抽象方法:

preHandle:控制器方法执行之前执行preHandle(),其boolean类型的返回值表示是否拦截或放行,返回true为放行,即调用控制器方法;返回false表示拦截,即不调用控制器方法

P82,P83:观察源码,preHandle如果返回true或false底层会怎么执行代码

postHandle:控制器方法执行之后执行postHandle()

afterComplation:处理完视图和模型数据,渲染视图完毕之后执行afterComplation()

3、多个拦截器的执行顺序 (日后很少用到,但强化思维)

a>若每个拦截器的preHandle()都返回true
此时多个拦截器的执行顺序和拦截器在SpringMVC的配置文件的配置顺序有关:
preHandle()会按照配置的顺序执行,
而postHandle()和afterComplation()会按照配置的反序执行

有一个index,只有一个prehandle()返回true,才会记录它的index

b>若某个拦截器的preHandle()返回了false(返回false还是会执行)
preHandle()返回false和它之前的拦截器的preHandle()都会执行,postHandle()都不执行,
返回false的拦截器之前的拦截器的afterComplation()会执行

二、异常处理器

1、基于配置的异常处理

SpringMVC提供了一个处理控制器方法执行过程中所出现的异常的接口:HandlerExceptionResolver

HandlerExceptionResolver接口的实现类有:DefaultHandlerExceptionResolver和SimpleMappingExceptionResolver

SpringMVC提供了自定义的异常处理器

我们执行其中的 doResolveException方法,处理异常,遇到异常的情况下,返回一个新的ModelAndView,可以渲染新的视图

SimpleMappingExceptionResolver,使用方式:

<bean class="org.springframework.web.servlet.handler.SimpleMappingExceptionResolver">
    <property name="exceptionMappings">
        <props>
        	<!--
        		properties的键表示处理器方法执行过程中出现的异常
        		properties的值表示若出现指定异常时,设置一个新的视图名称,跳转到指定页面
        	-->
            <prop key="java.lang.ArithmeticException">error</prop>
        </props>
    </property>
    <!--
    	exceptionAttribute属性设置一个属性名,将出现的异常信息在请求域中进行共享
    -->
    <property name="exceptionAttribute" value="ex"></property>
</bean>

2、基于注解的异常处理

@ExceptionHandler注解的属性value 是一个Class类型的数组,可以把出现数组里面类,映射到对应的页面

//@ControllerAdvice将当前类标识为异常处理的组件
@ControllerAdvice
public class ExceptionController {

    //@ExceptionHandler用于设置所标识方法处理的异常
    @ExceptionHandler(ArithmeticException.class)
    //ex表示当前请求处理中出现的异常对象
    public String handleArithmeticException(Exception ex, Model model){
        model.addAttribute("ex", ex);
        return "error";
    }
}

@RestControllerAdvice + @ExceptionHandler + 自定义返回工具类 是最优实战