Spring Data JPA – 在没有@Transactional的情况下获取了Lazy Loaded集合

我的期望是在事务范围内访问集合时应该获取延迟加载的集合。 例如,如果我想获取一个集合,我可以调用foo.getBars.size() 。 缺少活动事务应该导致exception,并显示错误消息

没有懒惰地初始化一个条形集合:….无法初始化代理 – 没有会话

但是,我注意到我的最新应用程序中的行为有所不同。 我正在使用Spring Boot 1.5.1和“data-jpa”启动器。 我过去使用过Spring Boot,但是数据jpa启动器对我来说是新的。

考虑以下情况。 我有一个懒惰加载的ManyToMany集合。

 @SuppressWarnings("serial") @Entity @Table(name = "foo") public class Foo implements java.io.Serializable { .... private Set bars = new HashSet(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 getBars() { return this.bars; } public void setBar(Set bars) { this.bars = bars; } 

我的服务方法未标记为Transactional,但我正在访问延迟加载的集合

 @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之后,我预计交易将被关闭。

这里发生了什么事?

Spring Boot默认使用OpenEntityManagerInView拦截器。 您可以通过将属性spring.jpa.open-in-view为false来将其关闭。

有关此(和其他)JPA属性的参考,请参阅文档 。

您可以打开日志记录以检查是否正在打开事务。

 org.hibernate.engine.transaction.internal.jdbc.JdbcTransaction 

要么

 org.hibernate.engine.transaction.internal.jta.JtaTransaction 

此外,您可以设置断点并使用此静态方法检查事务是否已打开。

 org.springframework.transaction.support.TransactionSynchronizationManager.isActualTransactionActive()