Spring Data JPA - 在没有 @Transactional 的情况下获取延迟加载的集合
Posted
技术标签:
【中文标题】Spring Data JPA - 在没有 @Transactional 的情况下获取延迟加载的集合【英文标题】:Spring Data JPA - Lazy Loaded collection fetched without @Transactional 【发布时间】:2017-07-23 21:25:23 【问题描述】:我的期望是在事务范围内访问集合时应该获取延迟加载的集合。例如,如果我想获取一个集合,我可以调用foo.getBars.size()
。没有活动事务应该会导致异常,并显示错误消息,例如
未能延迟初始化条形集合:....无法 初始化代理 - 没有会话
但是,我注意到我的最新应用程序中的行为有所不同。我正在使用带有“data-jpa”启动器的 Spring Boot 1.5.1。我过去使用过 Spring Boot,但 data-jpa 启动器对我来说是新的。
考虑以下情况。我有一个延迟加载的 ManyToMany 集合。
@SuppressWarnings("serial")
@Entity
@Table(name = "foo")
public class Foo implements java.io.Serializable
....
private Set<Bar> bars = new HashSet<Bar>(0);
....
@ManyToMany(cascade = CascadeType.ALL, fetch = FetchType.LAZY)
@JoinTable(name = "foo_bar_map",
joinColumns = @JoinColumn(name = "foo_id", nullable = false, updatable = false),
inverseJoinColumns = @JoinColumn(name = "bar_id", nullable = false, updatable = false))
public Set<Bar> getBars()
return this.bars;
public void setBar(Set<Bar> bars)
this.bars = bars;
我的服务方法未标记为事务性,但我正在访问延迟加载的集合
@Service
public class FooServiceImpl implements FooService
@Autowired
private FooRepository fooRepo;
@Override
public FooDTO findById(int fooId)
Foo foo = fooRepo.findOne(fooId);
// The FooDTO constructor will access foo.getBars()
return new FooDTO(foo);
对于 FooDTO 构造函数的上下文
public FooDTO(Foo foo)
...
for (Bar bar : foo.getBars())
this.bars.add(bar);
与我的期望和过去的经验相反,此代码成功执行并获取集合。此外,如果我在我的服务方法中抛出一个断点,我可以单步执行代码并查看我的日志中的 SQL 语句,这些语句在我调用 fooRepo 之后获取条形图。在我调用 fooRepo 之后,我希望交易会被关闭。
这里发生了什么?
【问题讨论】:
谁调用服务?我的猜测是调用者是事务性的。 @JBNizet 该方法由未标记为事务的控制器调用.... Spring boot 似乎默认使用 OpenEntityManagerInViewwInterceptor:github.com/spring-projects/spring-boot/blob/master/…。另请参阅docs.spring.io/spring-boot/docs/1.4.2.RELEASE/reference/…(并搜索 OpenEntityManagerInView) @JBNizet 感谢您挖掘它。这有效地回答了我的问题。如果作为答案发布,我会接受。 【参考方案1】:Spring Boot 默认使用 OpenEntityManagerInView 拦截器。您可以通过将属性 spring.jpa.open-in-view
设置为 false 来关闭它。
有关此(和其他)JPA 属性的参考,请参阅 the documentation。
【讨论】:
并非在所有情况下都会发生,为什么会这样?【参考方案2】:您可以打开日志记录以检查是否正在打开事务。
org.hibernate.engine.transaction.internal.jdbc.JdbcTransaction
或
org.hibernate.engine.transaction.internal.jta.JtaTransaction
另外,您可以设置断点并使用此静态方法来检查事务是否打开。
org.springframework.transaction.support.TransactionSynchronizationManager.isActualTransactionActive()
【讨论】:
【参考方案3】:你必须删除@JoinTable
【讨论】:
以上是关于Spring Data JPA - 在没有 @Transactional 的情况下获取延迟加载的集合的主要内容,如果未能解决你的问题,请参考以下文章
没有 JPQL 查询的 Spring-data-jpa 中的 CURRENT_DATE
spring data jpa getOne抛出LazyInitializationException而findBy没有
Spring Data JPA Update @Query 没有更新?
为啥在将 Spring Data JPA 与 Spring Boot 结合使用时,我的数据库自定义没有得到应用?
Spring Data JPA - 在没有 @Transactional 的情况下获取延迟加载的集合
@OneToMany Spring Boot 2.1.5.RELEASE 中的 Spring Data JPA 没有获取所有数据