繁体   English   中英

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

[英]Spring Data JPA - Lazy Loaded collection fetched without @Transactional

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

未能懒惰地初始化酒吧集合:.... 无法初始化代理 - 没有会话

但是,我注意到我最新的应用程序中的行为有所不同。 我将 Spring Boot 1.5.1 与“data-jpa”启动器一起使用。 我过去使用过 Spring Boot,但 data-jpa starter 对我来说是新的。

考虑以下情况。 我有一个懒加载的 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 之后,我预计交易将被关闭。

这里发生了什么事?

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

你必须删除@JoinTable

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM