Правильная конфигурация mvc: перехватчик в Spring
У меня проблема. Мне нужно вызвать каждый метод postHandle запроса в этом перехватчике:
public class DiMenuInterceptor extends HandlerInterceptorAdapter {
@Autowired
private IDiCategoryService categoryService;
@Override
public void postHandle(HttpServletRequest request,
HttpServletResponse response, Object handler,
ModelAndView modelAndView) throws Exception {
modelAndView.addObject("category", categoryService.getCategoryInTree());
}
}
поэтому я включил в сервлет конфигурации эти строки, и все работает нормально.
<bean class="org.springframework.web.servlet.mvc.annotation.DefaultAnnotationHandlerMapping" p:interceptors-ref="menuInterceptor" />
<bean id="menuInterceptor" class="cz.cosi.DiMenuInterceptor" />
Но теперь мне нужно изменить конфигурацию и использовать <mvc:interceptors>
В этой конфигурации я получаю серию исключений нулевого указателя в modelAndView в методе postHandle, потому что метод postHandle вызывается более одного раза за запрос.
<mvc:interceptors>
<bean class="cz.cosi.DiMenuInterceptor" />
</mvc:interceptors>
В этой конфигурации он работает, но только для запроса serverAdress/anything. Для запроса serverAdress/anything/something is postHandle не вызывается.
<mvc:interceptors>
<mvc:interceptor>
<mvc:mapping path="/*" />
<bean class="cz.cosi.DiMenuInterceptor" />
</mvc:interceptor>
</mvc:interceptors>
часть конфигурации сервлета
<mvc:annotation-driven />
<bean id="messageSource"
class="org.springframework.context.support.ReloadableResourceBundleMessageSource">
<property name="basename" value="classpath:messages" />
<property name="defaultEncoding" value="UTF-8" />
</bean>
<mvc:resources mapping="/css/**" location="/css/" />
<mvc:resources mapping="/images/**" location="/images/" />
<mvc:resources mapping="/js/**" location="/js/" />
<mvc:interceptors>
<bean class="cz.cosi.DiMenuInterceptor" />
</mvc:interceptors>
<tx:jta-transaction-manager />
<tx:annotation-driven />
<bean id="transactionManager"
class="org.springframework.orm.hibernate3.HibernateTransactionManager">
<property name="sessionFactory" ref="sessionFactory" />
</bean>
Кажется, что проблема может каким-то образом относиться к ресурсам, потому что с исключениями неправильно загружаются изображения, стили и javascript. Без mvc: ресурсы работают правильно, но это не решение.
Мой вопрос в том, как правильно настроить DiMenuInterceptor с помощью <mvc:interceptors>
. Большое спасибо за советы.
стек:
2011-04-14 09:56:02,487 [http-8080-3] DEBUG (FilterChainProxy.java:195) ? Converted URL to lowercase, from: '/images/core/users/super_admin.png'; to: '/images/core/users/super_admin.png'
2011-04-14 09:56:02,533 [http-8080-3] DEBUG (FilterChainProxy.java:202) ? Candidate is: '/images/core/users/super_admin.png'; pattern is /images/**; matched=true
2011-04-14 09:56:02,533 [http-8080-3] DEBUG (FilterChainProxy.java:158) ? /images/core/users/super_admin.png has an empty filter list
14.4.2011 9:56:02 org.apache.catalina.core.StandardWrapperValve invoke
SEVERE: Servlet.service() for servlet spring threw exception
java.lang.NullPointerException
at cz.cosi.DiMenuInterceptor.postHandle(DiMenuInterceptor.java:41)
at org.springframework.web.servlet.DispatcherServlet.doDispatch(DispatcherServlet.java:801)
at org.springframework.web.servlet.DispatcherServlet.doService(DispatcherServlet.java:719)
at org.springframework.web.servlet.FrameworkServlet.processRequest(FrameworkServlet.java:644)
at org.springframework.web.servlet.FrameworkServlet.doGet(FrameworkServlet.java:549)
at javax.servlet.http.HttpServlet.service(HttpServlet.java:617)
at javax.servlet.http.HttpServlet.service(HttpServlet.java:717)
at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:290)
at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:206)
at org.springframework.web.filter.CharacterEncodingFilter.doFilterInternal(CharacterEncodingFilter.java:88)
at org.springframework.web.filter.OncePerRequestFilter.doFilter(OncePerRequestFilter.java:76)
at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:235)
at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:206)
at org.springframework.security.web.FilterChainProxy.doFilter(FilterChainProxy.java:163)
at org.springframework.web.filter.DelegatingFilterProxy.invokeDelegate(DelegatingFilterProxy.java:237)
at org.springframework.web.filter.DelegatingFilterProxy.doFilter(DelegatingFilterProxy.java:167)
at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:235)
at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:206)
at org.apache.catalina.core.StandardWrapperValve.invoke(StandardWrapperValve.java:233)
at org.apache.catalina.core.StandardContextValve.invoke(StandardContextValve.java:191)
at org.apache.catalina.core.StandardHostValve.invoke(StandardHostValve.java:127)
at org.apache.catalina.valves.ErrorReportValve.invoke(ErrorReportValve.java:102)
at org.apache.catalina.core.StandardEngineValve.invoke(StandardEngineValve.java:109)
at org.apache.catalina.connector.CoyoteAdapter.service(CoyoteAdapter.java:298)
at org.apache.coyote.http11.Http11Processor.process(Http11Processor.java:859)
at org.apache.coyote.http11.Http11Protocol$Http11ConnectionHandler.process(Http11Protocol.java:588)
at org.apache.tomcat.util.net.JIoEndpoint$Worker.run(JIoEndpoint.java:489)
at java.lang.Thread.run(Thread.java:595)
Возможно, я нашел решение, но это определенно не самое лучшее. На данный момент кажется, что он работает.
<mvc:interceptors>
<mvc:interceptor>
<mvc:mapping path="/*" />
<bean class="cz.cosi.DiMenuInterceptor" />
</mvc:interceptor>
<mvc:interceptor>
<mvc:mapping path="/search/**" />
<bean class="cz.cosi.DiMenuInterceptor" />
</mvc:interceptor>
<mvc:interceptor>
<mvc:mapping path="/context/**" />
<bean class="cz.cosi.DiMenuInterceptor" />
</mvc:interceptor>
<mvc:interceptor>
<mvc:mapping path="/member/**" />
<bean class="cz.cosi.DiMenuInterceptor" />
</mvc:interceptor>
</mvc:interceptors>
Ответы
Ответ 1
Вам нужно указать, что ваш путь включает подпапки: /**"
вместо /*
.
<mvc:interceptors>
<mvc:interceptor>
<mvc:mapping path="/**" />
<bean class="cz.cosi.DiMenuInterceptor" />
</mvc:interceptor>
</mvc:interceptors>
@see Образец в Spring Ссылка, Глава 15.12.2 mvc: перехватчики
Ответ 2
Проблема заключается в том, что перехватчики вызываются при запросе ресурсов.
В сообщении говорится о том, как запретить вызывать перехватчики с помощью конфигурации xml. В принятом ответе я не большой поклонник полагаться на семантический состав путей (т.е. Используя .html или имеющую страницу на пути). Теперь все разработчики в команде должны знать об этом при создании контроллеров.
Я также не являюсь поклонником подробной конфигурации bean, поэтому я решил добавить следующий код к перехватчику:
if (ClassUtils.isAssignableValue(ResourceHttpRequestHandler.class, handler)) {
return;
}
// do interceptor logic here...
Ответ 3
У меня была очень простая проблема, но с доступом к шрифтам для Bootstrap 3. Перехватчики блокировали ресурсы шрифтов, такие как "glyphicons-halflings-regular.woff" или "glyphicons-halflings-regular.ttf", и они не были видны на первой странице.
Я разрешаю свою проблему следующим образом.
Определение моего перехватчика выглядит следующим образом:
<mvc:interceptors>
<mvc:interceptor>
<mvc:mapping path="/**"/>
<bean class="cz.cosi.DiMenuInterceptor"/>
</mvc:interceptor>
</mvc:interceptors>
И мой класс DiMenuInterceptor:
public class DiMenuInterceptor extends HandlerInterceptorAdapter {
private List<String> nonAuthPaths;
public DiMenuInterceptor() {
nonAuthPaths = new ArrayList<String>();
nonAuthPaths.add("/login");
nonAuthPaths.add("/resources/vendor/fonts/glyphicons-halflings-regular.woff");
nonAuthPaths.add("/resources/vendor/fonts/glyphicons-halflings-regular.ttf");
}
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
Object user = request.getSession().getAttribute(SessionManagement.USER_KEY);
if (user == null && !nonAuthPaths.contains(request.getServletPath())) {
response.sendRedirect(request.getContextPath() + "/login");
}
return false;
} else {
//ect ...
return true;
}
}
Ответ 4
У меня был тот же вопрос. Начальная конфигурация:
<mvc:interceptors>
<mvc:interceptor>
<mvc:mapping path="/*"/>
<bean class="cn.mmd.micro.common.TokenInterceptor">
<property name="excludeUrls">
<list>
<value>/app/token</value>
</list>
</property>
</bean>
</mvc:interceptor>
</mvc:interceptors>
Я изменил значение "mvc: mapping", и он сработал. Моя новая конфигурация выглядит следующим образом:
<mvc:interceptors>
<mvc:interceptor>
<mvc:mapping path="/**"/>
<bean class="cn.mmd.micro.common.TokenInterceptor">
<property name="excludeUrls">
<list>
<value>/app/token</value>
</list>
</property>
</bean>
</mvc:interceptor>