需要一些关于Spring自动装配和范围的帮助。
以下是基本的应用结构:
我有一个CustomHttpClient,注释为@Component,还从application.properties文件中提取一些与配置相关的属性(通过@Value注释)。
CustomHttpClient由我的应用程序中的多个服务使用。每当我使用CustomHttpClient时,我通过以下方式自动装配该实例:
@Autowired
private CustomHttpClient httpClient;
我使用拦截器来修改CustomHttpClient中的一些变量,如下所示:
public class MyInterceptor extends HandlerInterceptorAdapter {
@Autowired CustomHttpClient httpClient;
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
httpClient.setSomeProperty(newValue);
...
现在,这是问题所在。如果我按照上面的描述设置了所有内容,那么每当我通过拦截器更改CustomHttpClient的任何设置时,只要VM正在运行,就会为所有其他客户端保留该新值。因此,当我运行httpClient.setSomeProperty()时 - 该设置现在已永久保存。即使我从另一个客户端连接到该应用程序。
基本上我需要的是两件事:
我尝试将CustomHttpClient的范围更改为@Scope(“prototype”),但这样我就无法再使用拦截器更改CustomHttpClient的设置。
答案 0 :(得分:7)
默认情况下,当您使用@Autowired
时,spring bean范围是singleton。这意味着spring会使用@Autowired
注入相同的单例对象。通过创建范围prototype
,您指示Spring为每个@Autowired注入创建新对象,因此在您的拦截器中将拥有自己的HttpClient副本,并且无法查看其他HttpClient对象。
更好的方法是使用单例范围,使用请求属性或threadlocal在请求线程中随身携带自定义属性。即不是在拦截器中修改HttpClient属性,只需设置一些请求属性或threadlocals并在CustomHttpClient
类方法中处理这些自定义设置。
答案 1 :(得分:1)
如果你的拦截器只是添加一些属性,那么使用本地线程应该是更好的选择。您可以调用ThreadLocal.set(自定义Map)并在运行线程的任何位置使用它,当程序要离开控制器时,您可以调用ThreadLocal.Unset来清除存储的值。
这样你每次都不需要一个新的HttpcLient实例,每次都是一个新的实例将是一个严重的缺陷。并且您将能够在正在运行的线程中的任何位置使用自定义地图。
答案 2 :(得分:1)
通过XML或注释支持在Spring容器中声明的所有bean都是默认的单例。如果将具有设置为原型的范围的bean注入单个例如一个控制器,它只会注入一次。有一种方法可以实现这一目标。这是你应该如何声明作为原型的bean。这意味着每次从容器调用此bean时,容器将始终为您提供一个新实例。
<bean id="shoppingCart" class="example.ShoppingCart" scope="request">
<aop:scoped-proxy />
</bean>