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
我正在尝试实例化一个组件对象,我在 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