SpringMVC拦截器失效问题及解决方法

 更新时间:2024年01月06日 14:55:08   作者:哥的时代  
这篇文章主要介绍了SpringMVC 拦截器和异常处理器,以及SpringMVC拦截器失效分析解决方案,文中补充介绍了SpringMVC 拦截器和异常处理器的相关知识,需要的朋友可以参考下

SpringMVC拦截器失效

项目场景:

项目(springmvc)中新写了些拦截器没生效,记录一下原因

问题描述:

使用的xml的方式配置拦截器,具体如下

1.首先新建一个类实现HandlerInterceptor

public class LoginInterceptor implements HandlerInterceptor {
    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
        String sre = ".....";
        return false;
    }
    @Override
    public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
    }
    @Override
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
    }
}

2.接着在xml中注册拦截器

<mvc:interceptors>
        <mvc:interceptor>
            <mvc:mapping path="/apiweb/**"/>
            <bean class="com.xxx.interceptor.LoginInterceptor"/>
        </mvc:interceptor>
    </mvc:interceptors>

好像没什么大问题,实现 HandlerInterceptor接口,再将对应的类注册到容器中
但是,细节上存在很多问题,导致目前的配置中拦截器不生效

原因分析:

debug时发现根本就没有在容器中注册新写的拦截类,首先要做的是看看为什么没有注册成功后;之后再debug时返现url拦截路径不对,又修改了拦截路径。总共是两个问题,一个一个分析

在xml中配置<mvc:annotation-driven/>,从网上的信息看,这段代码会注册HandlerMapping和HandlerAdapter,在官方的解释中,有着更明确的回答

简单总结一下的话,添加了<mvc:annotation-driven/>后,就会自动注册自定义的HandlerInterceptors ,不再需要手动添加

debug一下

当不写<mvc:annotation-driven/>时,进行请求,在AbstractHandlerMapping类中

     容器中mappedInterceptors的个数为0

当使用<mvc:annotation-driven/>

容器中mappedInterceptors个数变成了2个,包含自定义的LoginInterceptor(图片中没显示出来。。。). 拦截器注册后,剩下拦截路径的配置
最初的拦截路径为

<mvc:mapping path="/apiweb/**"/>

这样写的原因是web.xml中的DispatcherServlet拦截的路径为/apiweb/**,以为拦截器这么写也没问题,但实际上路径错误导致拦截失败,debug发现,当输入的地址为http://localhost:8004/apiweb/test/a

lookupPath是/test/a,而拦截路径却是/apiweb/**,并不匹配,所以拦截路径也要进行修改,目前而言,改成/**即可

解决方案:

要修改的地方有两处:

  • 在xml配置文件中添加<mvc:annotation-driven/>
  • 将拦截器的拦截路径修改为/**

ps:如果使用注解配置的话,只需要添加一个类并继承WebMvcConfigurationSupport类,然后重写addInterceptors方法。如下

@Configuration
public class WebCon extends WebMvcConfigurationSupport {
   @Override
   public void addInterceptors(InterceptorRegistry registry) {
       registry.addInterceptor(new LoginInterceptor())
               .addPathPatterns("/**");
   }
}

看上去比xml配置简单很多,也不用操心<mvc:annotation-driven/>

补充:

SpringMVC 拦截器和异常处理器

一、拦截器

1、拦截器的配置

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

SpringMVC中的拦截器需要实现HandlerInterceptor

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

<bean class="com.gedeshidai.interceptor.FirstInterceptor"></bean>
<ref bean="firstInterceptor"></ref>
<!-- 以上两种配置方式都是对DispatcherServlet所处理的所有的请求进行拦截 -->
<mvc:interceptor>
    <mvc:mapping path="/**"/>
    <mvc:exclude-mapping path="/testRequestEntity"/>
    <ref bean="firstInterceptor"></ref>
</mvc:interceptor>
<!-- 
	以上配置方式可以通过ref或bean标签设置拦截器,通过mvc:mapping设置需要拦截的请求,通过mvc:exclude-mapping设置需要排除的请求,即不需要拦截的请求
-->

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

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

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

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

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

3、多个拦截器的执行顺序

a>若每个拦截器的preHandle()都返回true

此时多个拦截器的执行顺序和拦截器在SpringMVC的配置文件的配置顺序有关:

preHandle()会按照配置的顺序执行,而postHandle()和afterComplation()会按照配置的反序执行

b>若某个拦截器的preHandle()返回了false

preHandle()返回false和它之前的拦截器的preHandle()都会执行,postHandle()都不执行,返回false的拦截器之前的拦截器的afterComplation()会执行

二、异常处理器

1、基于配置的异常处理

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

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

SpringMVC提供了自定义的异常处理器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、基于注解的异常处理

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

总结

以上就是SpringMVC之拦截器和异常处理器的相关知识点,希望对你有所帮助。

到此这篇关于SpringMVC 拦截器和异常处理器的文章就介绍到这了,更多相关SpringMVC 拦截器内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Vue实现验证码登录的超详细步骤

    Vue实现验证码登录的超详细步骤

    这篇文章主要给大家介绍了关于Vue实现验证码登录的超详细步骤,我们在使用vue进行前端开发时都需要登录验证,文中通过代码示例介绍的非常详细,需要的朋友可以参考下
    2023-09-09
  • Java map.getOrDefault()方法的用法详解

    Java map.getOrDefault()方法的用法详解

    这篇文章主要介绍了Java map.getOrDefault()方法的用法详解,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-12-12
  • java编程调用存储过程中得到新增记录id号的实现方法

    java编程调用存储过程中得到新增记录id号的实现方法

    这篇文章主要介绍了java编程调用存储过程中得到新增记录id号的实现方法,涉及Java数据库操作中存储过程的相关使用技巧,需要的朋友可以参考下
    2015-10-10
  • Mybatis中 XML配置详解

    Mybatis中 XML配置详解

    这篇文章主要介绍了Mybatis中 XML配置详解的相关资料,非常不错,具有参考借鉴价值,需要的朋友可以参考下
    2017-01-01
  • 一文详解kafka序列化器和拦截器

    一文详解kafka序列化器和拦截器

    这篇文章主要为大家介绍了kafka序列化器和拦截器使用示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-03-03
  • Java线程中的ThreadLocal类解读

    Java线程中的ThreadLocal类解读

    这篇文章主要介绍了Java线程中的ThreadLocal类解读,ThreadLocal是一个泛型类,作用是实现线程隔离,ThreadLocal类型的变量,在每个线程中都会对应一个具体对象,对象类型需要在声明ThreadLocal变量时指定,需要的朋友可以参考下
    2023-11-11
  • SpringBoot2.x过后static下的静态资源无法访问的问题

    SpringBoot2.x过后static下的静态资源无法访问的问题

    这篇文章主要介绍了SpringBoot2.x过后static下的静态资源无法访问的问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-01-01
  • java 中Comparable与Comparator详解与比较

    java 中Comparable与Comparator详解与比较

    这篇文章主要介绍了java 中Comparable与Comparator详解与比较的相关资料,需要的朋友可以参考下
    2017-04-04
  • SpringBoot实现转页功能

    SpringBoot实现转页功能

    这篇文章主要介绍了SpringBoot实现转页功能,页面的跳转在web开发中是经常用的基础功能,感兴趣想要详细了解可以阅读下文,对大家的学习或工作具有一定的参考借鉴价值
    2023-05-05
  • Java中的volatile关键字解析

    Java中的volatile关键字解析

    这篇文章主要介绍了Java中的volatile关键字解析,Java内存模型规定了所有的变量都存储在主内存中,每个线程都有自己的工作内存,线程的工作内存保存了该线程使用到的变量的是主内存副本的拷贝,需要的朋友可以参考下
    2023-11-11

最新评论