Spring - 无法实例化请求范围内的组件对象

Spring - Unable to instantiate a component object in request scope

我正在尝试实例化一个组件对象,我在 applicationContext.xml 中为其声明了 <bean> 条目。实例化我的目标组件class的流程如下

计算控制器 -> 计算服务 -> 计算组件

在哪里

CalculatorController - scope = request,用@Controller 注释并包含在 web 中的组件扫描中applicationContext.xml

CalculatorService - Scope = singleton,用@service 注释并包含在 applicationContext.xml

的组件扫描中

CalculateComponent - scope = 请求,无注释,从 webapplicationConext.xml 和 applicationContext.xml 中的组件扫描中排除。在 webApplicationContext.xml 中定义的 bean 条目,范围 = request。还包括 <aop:scoped-proxy/><bean> 定义中。

我在 web.xml

中包含了以下条目
<context-param>
    <param-name>contextConfigLocation</param-name>
    <param-value>
            classpath:spring/applicationContext.xml
            /WEB-INF/mvc-dispatcher-servlet.xml
            ....Other resource xmls
    </param-value>
</context-param>

<listener>
    <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>

<!-- To retrieve session related information -->
<listener>
    <listener-class>org.springframework.security.web.session.HttpSessionEventPublisher</listener-class>
</listener>

<listener>
    <listener-class>
            org.springframework.web.context.request.RequestContextListener 
    </listener-class>
</listener> 


<servlet>
    <servlet-name>mvc-dispatcher</servlet-name>
    <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
    <load-on-startup>1</load-on-startup>
</servlet>

<servlet-mapping>
    <servlet-name>mvc-dispatcher</servlet-name>
    <url-pattern>/</url-pattern>
</servlet-mapping>

请注意,CalculateComponent 有一个 3 参数 ref 对象构造函数,它们全部三个在 webApplicationContext.xml 中都有 <bean> 条目,具有单例范围,并且它们没有注释。

当发送创建 CalculateComponent 对象的请求时,spring 容器抛出以下错误。

"No thread-bound request found: Are you referring to request attributes outside of an actual web request, or processing a request outside of the originally receiving thread? If you are actually operating within a web request and still receive this message, your code is probably running outside of DispatcherServlet/DispatcherPortlet: In this case, use RequestContextListener or RequestContextFilter to expose the current request."

请指教

更新:

当我从 contextConfigLocation 中删除 /WEB-INF/mvc-dispatcher-servlet.xml 并启动服务器时,我收到了 Autowired failed 错误 - "No qualifying bean of type CalculateComponent found for dependency:",即使我将范围从请求更改为单例。

然后我在 CalculateService 中注释掉了 CalculateComponent 的自动装配,现在我可以看到 CalculateComponent 启动了两次(正如我的@Serge Ballesta 提到的)。所以我得出结论,在加载 DispatcherServlet 之前(即没有加载 mvc-dispatcher-servlet 中提到的 bean),CalculateService 是通过 ContextLoaderListener(applicationContext.xml 中的 bean 条目)启动的。

我在 contextConfigLocation 中再次添加了 /WEB-INF/mvc-dispatcher-servlet.xml,但这次是作为第一个条目(即在 applicationContext.xml 之上)。现在 CalculateComponent 再次加载两次并且自动装配是在单例范围内完成的。通过此设置,我将 CalculateComponent 范围更改回请求,但再次出现 "I got the No thread-bound request found" 错误。

所以问题是,

ContextLoaderListener 试图在加载/可用之前初始化 DispatcherServlet(CalculateComponent) 的资源。

有一个请求范围的控制器很奇怪......据我所知,框架希望它是一个单例。尝试将其放在单例范围内。

而且我认为您已经将 /WEB-INF/mvc-dispatcher-servlet.xml 中的所有 bean 实例化了两次:

  • 首先在全局 contextConfigLocation 中声明的根应用程序上下文中
  • next 在名为 mvc-dispatcher
  • 的 servlet 的 servlet 应用程序上下文中

您应该重命名该文件,使其不作为 servlet 应用程序上下文加载器,或者将其从根上下文中删除

回答我自己的问题。正如我在问题中提到的,这是我获取请求范围组件 bean 的流程。

CalculateController -> CalculateService -> CalculateComponent。

但是 CalculateController 是通过异步请求调用的。 我们无法从异步请求线程访问 web 范围的 bean。

参考:How to enable request scope in async task executor