一尘不染

复制实体集合并坚持使用Hibernate / JPA

hibernate

我想在数据库中复制实体集合。我使用以下方法检索该集合:

CategoryHistory chNew = new CategoryHistory();
CategoryHistory chLast =  (CategoryHistory)em.createQuery("SELECT ch from CategoryHistory ch WHERE ch.date = MAX(date)").getSingleResult;
List<Category> categories = chLast.getCategories();
chNew.addCategories(categories)// Should be a copy of the categories: OneToMany

现在,我想复制“类别”列表,并使用EntityManager保留它。我正在使用JPA /hibernate。 更新

在知道如何分离我的实体之后,我需要知道要分离什么:当前代码:

    CategoryHistory chLast =  (CategoryHistory)em.createQuery("SELECT ch from CategoryHistory ch WHERE ch.date=(SELECT MAX(date) from CategoryHistory)").getSingleResult();
    Set<Category> categories =chLast.getCategories();

    //detach
    org.hibernate.Session session = ((org.hibernate.ejb.EntityManagerImpl) em.getDelegate()).getSession();
    session.evict(chLast);//detaches also its child-entities?

    //set the realations
    chNew.setCategories(categories);
    for (Category category : categories) {
        category.setCategoryHistory(chNew);
    }
    //set now create date
    chNew.setDate(Calendar.getInstance().getTime());

    //persist
    em.persist(chNew);

这引发了failed to lazily initialize a collection of role: entities.CategoryHistory.categories, no session or session was closed异常。

我认为他想重新加载类别,因为我已经将它们分离了。我现在该怎么办?


阅读 229

收藏
2020-06-20

共1个答案

一尘不染

亚伦·迪吉拉(AaronDiguila)的答案是去这里的方式,即您需要detach实例,将业务密钥设置为null,然后再将persist其设置为。

遗憾的是,无法使用JPA 1.x从实体管理器断开一个对象的连接(JPA
2.0将具有EntityManager.detach(Object)并修复此问题)。因此,要么等待JPA
2.x(我猜不是一个选项),要么使用Hibernate的底层Session

为此,您可以将EntityManager的委托转换为Hibernate Session。

Session session = (Session) em.getDelegate();

当然,这仅在将Hibernate用作Java Persistence提供程序时才有效,因为委托是会话API。

然后,分离对象:

session.evict(object);

更新:
根据在使用EntityManager.getDelegate()时要小心,对于GlassFish,应该实际使用(并且也可能在您的情况下使用):

org.hibernate.Session session = ((org.hibernate.ejb.EntityManagerImpl) em.getDelegate()).getSession();

但是,这会 不会 在JBoss中的工作,建议使用前面提到的代码。

org.hibernate.Session session = (Session) em.getDelegate();

虽然我知道使用getDelegate()会使JPA代码不可移植,但我必须承认我不希望此方法调用的结果特定于实现。

UPDATE2:
要回答问题的更新部分,我不确定您是否急于加载类别。这不是执行此操作的最佳方法,但是如果您categories.get(0)在驱逐之前致电,会发生什么?另外,我可能会缺少该部分,但是,您在哪里取消类别的键?

2020-06-20