SpingMVC第一个拦截器未执行BUG分析

问题描述:

SpringMvc项目中使用<mvc:interceptors>配置了一系列的拦截器,Web运行时发现第一个拦截器未执行。

 

问题分析:

1、第一个拦截器未注册上?

 

 

InterceptorsBeanDefinitionParser 中打断点,观察有生成对应的拦截器。

 

2、拦截器未命中?

 

AbstractHandlerMapping.getHandlerExecutionChain(Object handler, HttpServletRequest request)

 
发现规则命中,拦截器数量与MVC中配置数量一致,但是第一个拦截器确是ConversionServiceExposingInterceptor,应用中未配置该拦截器。
mappedInterceptors是在initApplicationContext()中初始化,跟踪代码进入如下片段:

 中间有一次递归调用,发现result来自两个BeanFactory,mvc中配置了四个拦截器,在此处跟踪发现在map中被覆盖,KEY值相同,为org.springframework.web.servlet.handler.MappedInterceptor#0,对应Class为ConversionServiceExposingInterceptor.class;
ConversionServiceExposingInterceptor在何处生成,为什么会有相同的BeanName,代码片段如下:

 

 String org.springframework.beans.factory.support.BeanDefinitionReaderUtils.generateBeanName(BeanDefinition definition, BeanDefinitionRegistry registry, boolean isInnerBean) throws BeanDefinitionStoreException

 


 

Generate a bean name for the given bean definition, unique within the given bean factory.

 

此代码只在一个bean工厂中保证名称唯一。

 

SpringMVC web.xml同时配置了ContextLoaderListener 和 DispatcherServlet,误将<mvc:interceptors>的配置放到了ContextLoaderListener 下,而DispatcherServlet解析<mvc:annotation-driven />时,默认注册ConversionServiceExposingInterceptor.class,生成了ID为org.springframework.web.servlet.handler.MappedInterceptor#0的拦截器,导致两个容器中的beanName重复,即处理请求时通过Map承载Interceptors的过程中导致第一个拦截器ConversionServiceExposingInterceptor覆盖。

 

 结论:涉及MVC相关的配置都要配置在DispatcherServlet中,不要一部分放到ContextLoaderListener,可能会因为名称冲突被覆盖。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
下面是一个完整的Spring MVC中多个拦截器的代码示例: 1. 创建拦截器类 `CustomInterceptor1.java`: ```java import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import org.springframework.web.servlet.HandlerInterceptor; import org.springframework.web.servlet.ModelAndView; public class CustomInterceptor1 implements HandlerInterceptor { @Override public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception { // 在请求处理之前进行拦截逻辑的处理 System.out.println("Custom Interceptor 1 - Pre Handle method is Calling"); return true; // 返回 true 表示继续执行后续的拦截器和请求处理方法,返回 false 则不再执行后续流程 } @Override public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception { // 在请求处理之后进行拦截逻辑的处理 System.out.println("Custom Interceptor 1 - Post Handle method is Calling"); } @Override public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception { // 在完成整个请求的处理后进行拦截逻辑的处理 System.out.println("Custom Interceptor 1 - Request and Response is completed"); } } ``` 2. 创建第二个拦截器类 `CustomInterceptor2.java`,代码与第一个拦截器类类似。 3. 配置拦截器 `springmvc-servlet.xml`: ```xml <!-- 配置拦截器 --> <mvc:interceptors> <mvc:interceptor> <mvc:mapping path="/**"/> <!-- 拦截所有的请求 --> <bean class="com.example.CustomInterceptor1"/> </mvc:interceptor> <mvc:interceptor> <mvc:mapping path="/**"/> <!-- 拦截所有的请求 --> <bean class="com.example.CustomInterceptor2"/> </mvc:interceptor> </mvc:interceptors> ``` 请确保将 `com.example.CustomInterceptor1` 和 `com.example.CustomInterceptor2` 替换为你实际的拦截器类的包名和类名。 在以上示例中,我们创建了两个拦截器类 `CustomInterceptor1` 和 `CustomInterceptor2`,并分别实现了 `HandlerInterceptor` 接口。在配置文件中,我们使用 `<mvc:interceptors>` 标签来配置多个拦截器,并使用 `<mvc:mapping>` 标签指定要拦截的请求路径。然后,我们将拦截器类的 bean 配置在 `<bean>` 标签中。 多个拦截器会按照配置顺序依次执行,并且在请求处理前和处理后可以执行相应的拦截逻辑。 你可以根据自己的需求和业务逻辑,配置任意多个拦截器,并调整它们的顺序。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值