由于doubled上下文(servlet + ContextLoaderListener),所有Spring Framework bean都会重复

时间:2014-03-16 11:25:12

标签: java spring spring-mvc servlets

  • 如果我通过调度程序servlet创建弹簧上下文 ,我在DelegatingFilterProxy过滤器中出错:

    java.lang.IllegalStateException: No WebApplicationContext found: no ContextLoaderListener registered?
    org.springframework.web.filter.DelegatingFilterProxy.doFilter(DelegatingFilterProxy.java:251)
    org.springframework.web.filter.CharacterEncodingFilter.doFilterInternal(CharacterEncodingFilter.java:88)
    org.springframework.web.filter.OncePerRequestFilter.doFilter(OncePerRequestFilter.java:107)
    org.apache.logging.log4j.core.web.Log4jServletFilter.doFilter(Log4jServletFilter.java:66)
    
  • 如果我按ContextLoaderListener 创建弹簧上下文,我只有{{1>}错误,因为没有servlet

  • 如果我通过servlet和&创建弹簧上下文监听器我有重复的上下文,因此所有bean都是重复的,包括带有请求映射的控制器,双重执行404方法等。

如何创建高级弹簧应用程序(包括大量过滤器等)而不重复上下文?

我的web.xml:

@Scheduled

1 个答案:

答案 0 :(得分:7)

如果您使用与弹簧相关的servlet过滤器并且还使用mvc控制器,那么您需要:

  • ContextLoaderListener AND
  • DispatcherServlet的弹簧构型

(见ContextLoaderListener or not?

两者都创建自己的servlet上下文。 ContextLoaderListener创建父上下文(有时称为内部上下文)。 DispatcherServlet创建一个子上下文(父上下文)(有时称为外部上下文)。 子上下文的bean可以访问父上下文的bean,但不能反过来。

在一个不太简单的Web应用程序中,您需要两个上下文,因为有许多servlet过滤器需要已经创建的spring上下文。另一方面,所有控制器东西都需要一个ServletContext,而这只是由Dispatcher Servlet创建的。

另一点是,你不应该每次创建两次bean(有时这不是问题,有时候也是如此)。所以你需要有两个弹簧配置,一个用于内部上下文,一个用于另一个上下文。而且你需要为每个bean决定它是属于内部还是外部。

经验法则是:将所有内容放在内部上下文中,除了那些需要Servlet上下文或者与Web前端密切相关的东西,比如MVC-Controllers,Tiles配置,....