一尘不染

分离具有延迟初始化属性的JPA对象

hibernate

有两个JPA实体:具有一对多关系的用户和订单。

/**
 * User DTO
 */
@Entity
@Table(name="user")
public class User implements Serializable {
    private static final long serialVersionUID = 8372128484215085291L;

    private Long id;
    private Set<Order> orders;

    public User() {}

    @Id @GeneratedValue(strategy=GenerationType.SEQUENCE, generator="sequenceUser")
    public Long getId() {
        return this.id;
    }
    private void setId(Long id) {
        this.id = id;
    }

    @OneToMany(mappedBy="user", cascade=CascadeType.PERSIST, fetch=FetchType.LAZY)
    @LazyCollection(LazyCollectionOption.EXTRA)
    public Set<Order> getOrders() {
        return orders;
    }
    public void setOrders(Set<Order> orders) {
        this.orders = orders;
    }
 }


/**
 * Order DTO
 */
@Entity
@Table(name="order")
public class Order implements Serializable {
    private static final long serialVersionUID = 84504362507297200L;

    private Long id;
    private User user;

    public Order() {
    }

    @Id @GeneratedValue(strategy=GenerationType.SEQUENCE, generator="sequenceOrder")
    public Long getId() {
        return this.id;
    }
    private void setId(Long id) {
        this.id = id;
    }

    @ManyToOne
    @JoinColumn(name="user_id")
    public User getUser(){
        return user;
    }
    public void setUser(User user){
        this.user = user;
    }
}

我在每个方法都在事务中运行的服务层类中使用这些实体。一切都很好,除非服务层类的方法必须返回这些实体。

@Transactional(readOnly=true)
public Set<Order> getOrders() {
    Set<Order> orders = user.getOrders();

    return orders;
}

此方法可以很好地返回数据。但是,当我尝试访问接收到的集合元素时,我捕获到异常:“
org.hibernate.LazyInitializationException:无法延迟初始化角色集合:package.User.orders,没有会话或会话被关闭”。

因此,它被排除在外。我以为分离结果会解决我的问题,但是这样的把戏

@Transactional(readOnly=true)
public Set<Order> getOrders() {
    Set<Order> orders = user.getOrders();

    for(Order order: orders)
        entityManager.detach(order);
    return orders;
}

没有任何改变:(

对我而言,是否将有关用户是否参加订单集的信息无关紧要。我只想使用此集合,而不要对其进行修改。

有谁能够帮助我?:)


阅读 198

收藏
2020-06-20

共1个答案

一尘不染

此方法可以很好地返回数据。但是,当我尝试访问接收到的集合元素时,我捕获到异常:“
org.hibernate.LazyInitializationException:无法延迟初始化角色集合:package.User.orders,没有会话或会话被关闭”。

该错误是自我解释:您正尝试加载(延迟)加载的集合,但由于User实例已分离,因此不再有活动会话。

因此,它被排除在外。我以为分离结果会解决我的问题,但是这样的把戏

这不会改变任何东西。该EntityManager#detach(Object)方法不
加载 任何内容,它从持久性上下文中删除传递的实体,使其成为 detached

对我而言,是否将有关用户是否参加订单集的信息无关紧要。我只想使用此集合,而不要对其进行修改。

您需要建立关联EAGER(以便在检索用户时加载订单),或者在服务中检索用户时使用FETCH JOIN:

SELECT u
FROM User u LEFT JOIN FETCH u.orders
WHERE u.id = :id

我是否正确理解hibernate没有强制加载当前对象的所有惰性关联的标准机制

Hibernate有一个Hibernate.initialize方法,但这显然不是标准的JPA(对于可移植代码,首选fetch join)。

我没有办法让hibernate状态忽略当前对象的惰性关联(将它们设置为null)

什么?忽略是什么意思?你为什么要那样做?

2020-06-20