Spring是RestTemplate
线程安全的吗?那是
- 是
RestTemplate
可以安全共享多个连接的策略对象。要么 - 是一个
RestTemplate
连接对象(如数据库连接),使用时无法共享,并且需要为每个连接重新创建或池化。
Spring是RestTemplate
线程安全的吗?那是
RestTemplate
可以安全共享多个连接的策略对象。要么RestTemplate
连接对象(如数据库连接),使用时无法共享,并且需要为每个连接重新创建或池化。Answers:
RestTemplate
是线程安全的(添加了重点):
从概念上讲,它是非常相似的
JdbcTemplate
,JmsTemplate
和Spring框架和其他投资项目中发现的各种其他模板。举例来说,这意味着RestTemplate
一旦构建后,线程安全
RestTemplate
该类的对象不会更改其任何状态信息来处理HTTP:该类是Strategy设计模式的一个实例,而不是像一个连接对象。没有状态信息,如果它们共享一个RestTemplate
对象,就不可能有不同的线程破坏或加速状态信息。这就是为什么线程可以共享这些对象的原因。
如果检查源代码,RestTemplate
您将发现在构造对象之后,它不使用synchronized
方法或volatile
字段来提供线程安全。因此,它是不是安全的修改RestTemplate
施工后的对象。特别是,添加消息转换器是不安全的。
要为它提供消息转换器列表,您必须执行以下操作之一:
RestTemplate(List<HttpMessageConverter<?>> messageConverters)
构造函数。作为messageConverters
is的内部列表final
,它安全地发布了消息转换器的列表。setMessageConverters(List<HttpMessageConverter<?>> messageConverters)
增变器,然后安全地发布更改的RestTemplate
对象。使用具有此功能的Spring bean定义<property name="messageConverters"><list>...
,因为在大多数实际使用情况下,将通过设置容器的线程安全地发布bean 。List.add
在返回的引用上使用getMessageConverters()
,然后安全地发布更改的RestTemplate
对象。但是,for的文档RestTemplate
未明确声明其返回了可用于更改消息转换器列表的引用。当前实现可以,但是可能实现可以更改为返回Collections.unmodifiableList
列表的a或副本。因此,最好不要以这种方式进行更改。需要注意的是第一种情况是构建对象时建立的消息转换器的唯一手段,因此它是正确的说,它“是线程安全的,一旦构成”。
该类是Spring框架的一部分,因此在几乎所有实际情况下,该类的对象都将被设置为Spring Application Context的一部分,使用第一个(使用构造函数的依赖注入)或第二个(使用setter的依赖注入)方法,因此可以保证安全地发布到多个线程。
好的,尽管我可能会从导致这些问题的源代码管理中挖掘出旧代码。
我认为可以说,即使在创建时进行同步,也存在其他线程可以修改内部集合的情况。所以最好小心。查看旧代码,是的,它实际上是在使用消息转换器。但仅在创建时同步。
restTemplate = new RestTemplate();
restTemplate.getMessageConverters().add(new MappingJackson2HttpMessageConverter());
之后,与RestTemplate的唯一交互是与此:
return restTemplate.postForObject(url, object, clazz);
这也是最终引发异常的行。
当然,与消息转换器没有任何交互(我们没有本地引用)。
查看stacktrace和spring源代码,在此行发生了错误:
for (HttpMessageConverter<?> converter : getMessageConverters()) {
那我们有什么呢?
因此,总而言之,在某些情况下,事情可能不是线程安全的,当然,如果您直接使用消息转换器的话。这种情况虽然很奇怪,但是我认为发布它会很有用。
restTemplate.getMessageConverters().add(new MappingJackson2HttpMessageConverter());
,请不要安全地发布对所做的更改restTemplate
。
restTemplate.getMessageConverters().add(...
不是以线程安全的方式进行的。
讨厌不同意上面接受的答案(添加了重点),但是不,这不是线程安全的。即使在创建之后。在内部它正在与ArrayLists一起玩,我还没有深入研究源代码。我看过太多这些:
java.util.ConcurrentModificationException
at java.util.ArrayList$Itr.checkForComodification(ArrayList.java:859)
at java.util.ArrayList$Itr.next(ArrayList.java:831)
at org.springframework.web.client.RestTemplate$AcceptHeaderRequestCallback.doWithRequest(RestTemplate.java:677)
at org.springframework.web.client.RestTemplate.doExecute(RestTemplate.java:567)
at org.springframework.web.client.RestTemplate.execute(RestTemplate.java:545)
at org.springframework.web.client.RestTemplate.getForObject(RestTemplate.java:253)
RestTemplate
。异常为aConcurrentModificationException
并不表示一旦构造RestTemplate
后就不是线程安全的。这与您在构造对象之后更改对象或其包含的对象之一(例如消息转换器列表)的代码一致。RestTemplate
ArrayList