问题描述
RestTemplate
是线程安全的(添加了重点):
从概念上讲,它是非常相似的JdbcTemplate
,JmsTemplate
和Spring框架和其他投资项目中发现的各种其他模板。举例来说,这意味着RestTemplate
一旦构建后,线程安全
RestTemplate
该类的对象不会更改其任何状态信息来处理HTTP:该类是Strategy设计模式的一个实例,而不是像一个连接对象。没有状态信息,如果它们共享一个RestTemplate
对象,就不可能有不同的线程破坏或加速状态信息。这就是为什么线程可以共享这些对象的原因。
如果检查源代码,RestTemplate
你将发现在构造对象之后,它不使用synchronized
方法或volatile
字段来提供线程安全。因此,它是不是安全的修改RestTemplate
施工后的对象。特别是,添加消息转换器是不安全的。
要为它提供消息转换器列表,你必须执行以下操作之一:
- 使用
RestTemplate(List<HttpMessageConverter<?>> messageConverters)
构造函数。作为messageConvertersis
的内部列表final,它安全地发布了消息转换器的列表。 - 使用
setMessageConverters(List<HttpMessageConverter<?>> messageConverters)
增变器,然后安全地发布更改的RestTemplate
对象。使用具有此功能的Spring bean定义<property name="messageConverters"><list>...
,因为在大多数实际使用情况下,线程会通过设置容器的线程安全地发布该bean 。 -
List.add
在返回的引用上使用getMessageConverters()
,然后安全地发布更改的RestTemplate
对象。但是,for的文档RestTemplate
未明确声明它返回了可用于更改消息转换器列表的引用。当前实现可以,但是可能实现可以更改为返回Collections.unmodifiableList
列表的一个或副本。因此,最好不要以这种方式进行更改。 需要注意的是第一种情况是构建对象时建立的消息转换器的唯一手段,因此它是正确的说,它“是线程安全的,一旦构成”。
该类是Spring框架的一部分,因此在几乎所有实际情况下,该类的对象都将被设置为Spring Application Context的一部分,使用第一个(使用构造函数的依赖注入)或第二个(使用setter的依赖注入)方法,因此可以保证安全地发布到多个线程。
解决方法
Spring是RestTemplate
线程安全的吗?那是
- 是
RestTemplate
可以安全共享多个连接的策略对象。 - 是一个
RestTemplate
连接对象(如数据库连接),使用时无法共享,并且需要为每个连接重新创建或池化。