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()