我正在尝试实例化一个组件对象,我在applicationContext.xml中声明了<bean>
条目。实例化我的目标组件类的流程如下
CalculateController -> CalculateService -> CalculateComponent
,
CalculatorController - scope = request,用@Controller注释并包含在webapplicationContext.xml中的组件扫描中
CalculatorService - Scope = singleton,用@service注释并包含在applicationContext.xml中的组件扫描中
CalculateComponent - scope =请求,没有注释,在webapplicationcontext .xml和applicationContext.xml中被排除在组件扫描之外。在webApplicationContext.xml中定义了范围= request的bean条目。还包括<bean>
定义中的<aop:scoped-proxy/>
我在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容器抛出以下错误。
"没有找到线程绑定的请求:您是否指的是请求。属性在实际的web请求之外,或者正在处理请求在最初接收线程之外?如果你是在一个网络请求内操作,仍然收到此消息,您的代码可能在外部运行DispatcherServlet/DispatcherPortlet:在本例中,使用RequestContextListener或RequestContextFilter公开当前请求。"
请建议。
更新:
当我从contextConfigLocation中删除/WEB-INF/mvc-dispatcher-servlet.xml并启动服务器时,我得到了Autowired失败的错误-"没有为依赖项找到CalculateComponent类型的合格bean:",即使在我将范围从请求更改为单例之后。
然后我在CalculateService中注释掉了CalculateComponent的自动生成,现在我可以看到CalculateComponent被启动了两次(正如我的@Serge Ballesta提到的)。因此,我得出结论,在DispatcherServlet被加载之前,CalculateService是通过ContextLoaderListener (applicationContext.xml中的bean条目)初始化的。没有加载mvc-dispatcher-servlet中提到的bean)。
我再次在contextConfigLocation中添加了/WEB-INF/mvc-dispatcher-servlet.xml,但这次作为第一个条目(即在applicationContext.xml之上)。现在,CalculateComponent再次加载了两次,自动装配在单例范围内完成了。通过这个设置,我将CalculateComponent范围更改回request,但再次出现"I got the No thread-bound request found"错误。
问题是,
ContextLoaderListener试图在DispatcherServlet(CalculateComponent)加载/可用之前初始化其资源。
这是奇怪的有一个请求范围的控制器…据我所知,框架希望它是单例的。试着把它放在单例作用域中。
并且我认为您已经将/WEB-INF/mvc-dispatcher-servlet.xml
中的所有bean实例化了两次:
- 首先在根应用上下文中声明,因为它是在全局contextConfigLocation 中声明的
- 在名为
mvc-dispatcher
的servlet应用程序上下文中
您应该重命名文件,这样它就不会被加载为servlet应用程序上下文,或者从根上下文中删除它
回答我自己的问题。正如我在问题中提到的,这是获取请求作用域组件bean的流程。
CalculateController -> CalculateService -> CalculateComponent
但是CalculateController是通过异步请求调用的。我们不能从异步请求线程访问web作用域bean
参考:如何在异步任务执行器中启用请求作用域