Hibernate在创建SessionFactory时抛出这个异常:

multiplebagfetchexception:不能同时获取多个包

这是我的测试用例:

Parent.java

@Entity
public Parent {

 @Id
 @GeneratedValue(strategy=GenerationType.IDENTITY)
 private Long id;

 @OneToMany(mappedBy="parent", fetch=FetchType.EAGER)
 // @IndexColumn(name="INDEX_COL") if I had this the problem solve but I retrieve more children than I have, one child is null.
 private List<Child> children;

}

Child.java

@Entity
public Child {

 @Id
 @GeneratedValue(strategy=GenerationType.IDENTITY)
 private Long id;

 @ManyToOne
 private Parent parent;

}

这个问题怎么样?我该怎么办?


EDIT

好的,我的问题是,另一个“父”实体是在我的父,我的真实行为是这样的:

Parent.java

@Entity
public Parent {

 @Id
 @GeneratedValue(strategy=GenerationType.IDENTITY)
 private Long id;

 @ManyToOne
 private AnotherParent anotherParent;

 @OneToMany(mappedBy="parent", fetch=FetchType.EAGER)
 private List<Child> children;

}

AnotherParent.java

@Entity
public AnotherParent {

 @Id
 @GeneratedValue(strategy=GenerationType.IDENTITY)
 private Long id;

 @OneToMany(mappedBy="parent", fetch=FetchType.EAGER)
 private List<AnotherChild> anotherChildren;

}

Hibernate不喜欢FetchType有两个集合。EAGER,但这似乎是一个bug,我没有做不寻常的事情…

删除FetchType。来自Parent或AnotherParent的渴望解决了这个问题,但我需要它,所以真正的解决方案是使用@LazyCollection(LazyCollectionOption.FALSE)而不是FetchType(感谢Bozho的解决方案)。


当前回答

当你有一个太复杂的对象和一个简单的收集不是一个好主意,所有的对象都用EAGER fetchType,最好使用LAZY,当你真的需要加载集合使用:Hibernate.initialize(parent.child)来获取数据。

其他回答

要解决这个问题,只需将Set替换为嵌套对象的List。

@OneToMany
Set<Your_object> objectList;

不要忘记使用fetch=FetchType。急切的

它会起作用的。

如果你想只使用list, Hibernate中还有一个概念CollectionId。

但请记住,你不会消除Vlad Mihalcea在他的回答中所描述的底层笛卡尔积!

您可以在JPA中保留摊位EAGER列表,并向其中至少一个添加JPA注释@OrderColumn(显然是要排序的字段的名称)。不需要特定的hibernate注释。 但是请记住,如果所选字段没有从0开始的值,它可能会在列表中创建空元素

 [...]
 @OneToMany(mappedBy="parent", fetch=FetchType.EAGER)
 @OrderColumn(name="orderIndex")
 private List<Child> children;
 [...]

在Children中,你应该添加orderIndex字段

在尝试了这篇文章和其他文章中描述的每一个选项后,我得出的结论是,解决方案如下。

在每个XToMany地方@XXXToMany(mappedBy="parent", fetch=FetchType.EAGER) 在中间之后

@Fetch(value = FetchMode.SUBSELECT)

这对我很有效

我认为hibernate的新版本(支持JPA 2.0)应该可以处理这个问题。但除此之外,你可以通过注释集合字段来解决它:

@LazyCollection(LazyCollectionOption.FALSE)

记住要从@*ToMany注释中删除fetchType属性。

但请注意,在大多数情况下,Set<Child>比List<Child>更合适,所以除非你真的需要List -选择Set

但请记住,使用集合时,你不会消除Vlad Mihalcea在他的回答中所描述的底层笛卡尔积!

你可以使用一个新的注释来解决这个问题:

@XXXToXXX(targetEntity = XXXX.class, fetch = FetchType.LAZY)

实际上,fetch的默认值是FetchType。懒惰。