无法在请求范围内实例化组件对象



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

参考:如何在异步任务执行器中启用请求作用域

最新更新