我想在数据库中复制实体集合。我使用以下方法检索该集合:
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异常。
failed to lazily initialize a collection of role: entities.CategoryHistory.categories, no session or session was closed
我认为他想重新加载类别,因为我已经将它们分离了。我现在该怎么办?
亚伦·迪吉拉(AaronDiguila)的答案是去这里的方式,即您需要detach实例,将业务密钥设置为null,然后再将persist其设置为。
detach
null
persist
遗憾的是,无法使用JPA 1.x从实体管理器断开一个对象的连接(JPA 2.0将具有EntityManager.detach(Object)并修复此问题)。因此,要么等待JPA 2.x(我猜不是一个选项),要么使用Hibernate的底层Session。
EntityManager.detach(Object)
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代码不可移植,但我必须承认我不希望此方法调用的结果特定于实现。
getDelegate()
UPDATE2: 要回答问题的更新部分,我不确定您是否急于加载类别。这不是执行此操作的最佳方法,但是如果您categories.get(0)在驱逐之前致电,会发生什么?另外,我可能会缺少该部分,但是,您在哪里取消类别的键?
categories.get(0)