Spring Data JPA - 在没有 @Transactional 的情况下获取懒加载的集合

10

我的期望是:当在事务范围内访问一个懒加载的集合时,应该取回该集合。例如,如果我想取回一个集合,可以调用foo.getBars.size()。如果没有活动的事务,则应该返回一个异常,并显示错误信息如下:

failed to lazily initialize a collection of bars: .... could not initialize proxy - no Session

然而,我注意到在我最新的应用程序中行为不同。我正在使用Spring Boot 1.5.1和"data-jpa"启动器。我以前使用过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;
    }

我的服务方法没有标记为“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);
    }
}

与我的期望和过去的经验相反,该代码成功执行并获取了集合。此外,如果我在服务方法中设置断点,我可以逐步执行代码,并在日志中看到在调用fooRepo后获取bar的SQL语句。在调用fooRepo后,我希望事务被关闭。

这里发生了什么?


谁调用了服务?我猜想是事务性的调用者。 - JB Nizet
@JBNizet 这个方法是被一个没有标记为事务的控制器调用的... - The Gilbert Arenas Dagger
1
Spring Boot默认似乎使用OpenEntityManagerInViewwInterceptor: https://github.com/spring-projects/spring-boot/blob/master/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/orm/jpa/JpaBaseConfiguration.java#L203。参见http://docs.spring.io/spring-boot/docs/1.4.2.RELEASE/reference/htmlsingle/#common-application-properties(搜索OpenEntityManagerInView)。 - JB Nizet
@JBNizet 感谢您的努力查找。这有效地回答了我的问题。如果发布为答案,我会接受它。 - The Gilbert Arenas Dagger
3个回答

20

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

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


1
并非所有情况下都会发生,为什么会这样? - Alpit Anand

3

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

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

或者

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

此外,您可以设置断点并使用此静态方法来检查是否已打开事务。
org.springframework.transaction.support.TransactionSynchronizationManager.isActualTransactionActive()

0

你必须移除 @JoinTable


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接