当前位置: 首页 > 面试题库 >

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

钱安和
2023-03-14
问题内容

有两个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;
}

没有任何改变:(

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

有谁能够帮助我?:)


问题答案:

此方法可以很好地返回数据。但是,当我尝试访问接收到的集合元素时,我捕获到异常:“
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)

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



 类似资料:
  • 我想在Spring靴中通过kotlin在另一个类中注入一个单例。 S.kt WorkingGroup.kt 通过这段代码,我得到以下错误: 我搜索了这个错误,发现了一些类似的结果,但是问题没有解决。< br >在kotlin中,如何在伴随对象中注入服务?

  • 这是我的主要功能 我在我的主类中调用它,并且一切正常,我可以得到响应。 当我想为它编写测试时,我遇到了错误“kotlin.UninitializedPropertyAccessException:lateinit属性客户端尚未初始化”,有人能帮我吗?"

  • 7.4.4 延迟初始化的bean 默认情况下,ApplicationContext实现在初始化过程中随即创建和配置所有单例bean。一般来说,这种预实例化是可取的,因为可以立即发现配置或周围环境中的错误,而不是在几个小时甚至几天以后。当这种行为不可取时,可以通过将bean定义标记为延迟初始化来阻止预实例化。延迟初始化的bean告诉IoC容器,当bean首次被请求时而不是在启动时创建一个实例。 在X

  • 本文向大家介绍Python性能提升之延迟初始化,包括了Python性能提升之延迟初始化的使用技巧和注意事项,需要的朋友参考一下 所谓类属性的延迟计算就是将类的属性定义成一个property,只在访问的时候才会计算,而且一旦被访问后,结果将会被缓存起来,不用每次都计算。构造一个延迟计算属性的主要目的是为了提升性能 property 在切入正题之前,我们了解下property的用法,property可

  • 有人以前有过类似的问题吗? 如何声明init()的默认变量值? 下面是我的代码示例, 然后在下面抛出异常: 起因:科特林。UninitializedPropertyAccessException:late init属性emailDir尚未初始化 任何解决方案都可以共享?

  • 问题内容: 在我的JavaFX应用程序中,当我对EmbeddedId类使用“ 非延迟收集” 选项时,为了保持双向关系,它将产生以下异常。经过几个小时的工作后,我仍然无法解决它。实体类如下。谢谢。 EmbeddedId类 WorkflowStep实体 (带有EmbeddedId) WfScriptTemplate实体 (具有双向关系) PostgreSQL异常 MySQL异常 问题答案: 摘录自JP