【问题标题】:Inheritance JPA and Hibernate Issue继承 JPA 和 Hibernate 问题
【发布时间】:2014-06-30 10:52:22
【问题描述】:

我在加载一些对象时遇到了一个奇怪的问题。我正在使用 JPA 1、hibernate-core 版本 3.3.0.SP1 和 hibernate-entitymanager 版本 3.4.0.GA

假设我有这些 JPA 实体:

@Entity
@Table(name = "SLC_ELE")
@Inheritance(strategy = InheritanceType.JOINED)
@DiscriminatorColumn(discriminatorType = DiscriminatorType.INTEGER, name = ElementoPrograma.C_ID_CTG_ELE)
public class Element {
...
} 

@Entity
@Table(name = "SLC_ELE_ONE")
@Inheritance(strategy = InheritanceType.JOINED)
@DiscriminatorValue(Categories.ID_CTG_ONE)
public class ElementTypeOne extends Element {
    ...
}

@Entity
@Table(name = "SLC_ELE_TWO")
@Inheritance(strategy = InheritanceType.JOINED)
@DiscriminatorValue(Categories.ID_CTG_TWO)
public class ElementTypeTwo extends Element {
    ...
} 

@Entity
@Table(name = ThreeElementExample.TABLENAME)
@AssociationOverrides({
    @AssociationOverride(name = JpaMany3ManyEntity.ASOCIATION_OVERRIDE_ONE,
    joinColumns =
    @JoinColumn(name = Element.C_ID_ELE)),
    @AssociationOverride(name = JpaMany3ManyEntity.ASOCIATION_OVERRIDE_TWO,
    joinColumns =
    @JoinColumn(name = OneEntity.C_ID_TWO)),
    @AssociationOverride(name = JpaMany3ManyEntity.ASOCIATION_OVERRIDE_THREE,
    joinColumns =
    @JoinColumn(name = AnotherEntity.C_ID_THREE))}) 
public class ThreeElementExample extends JpaMany3ManyEntity<Element, OneEntity, AnotherEntity> {
 ...
}

问题是,当我加载这些实体的集合时,我想始终获取子类(意思是 ElementTypeOne、ElementTypeTwo 而不是元素)。问题是多对多关系总是得到元素(父亲而不是孩子)

假设我有一个包含元素集合的实体 A:

@Fetch(FetchMode.JOIN)
@OneToMany(cascade = CascadeType.ALL, mappedBy = "idEle")
private Collection<Element> elementCollection;

如果我得到集合,一切正常(我得到了预期的子类)。

当我有另一个带有 JpaMany3ManyEntity 集合的实体 B 时出现问题(注意涉及到相同的实体元素)

@OneToMany(cascade = CascadeType.ALL, mappedBy = JpaMany3ManyEntity.ASOCIATION_OVERRIDE_ONE, fetch = FetchType.LAZY)
private Collection<ThreeElementExample> threeElementExampleCollection;

如果我在尝试从 A 类获取 elementCollection 之前从 B 类循环了 threeElementExampleCollection,当我从 elementCollection 加载对象时 我只获得超类(元素)对象而不是孩子。

我想,出于任何原因,多对多关系总是获取 Element 对象(父亲)并将它们保存在休眠缓存中,但我需要避免这种行为。

有什么想法或解决方法吗?任何形式的帮助将不胜感激。

提前致谢。

编辑:多对多类:

@SuppressWarnings("serial")
@MappedSuperclass
@AssociationOverrides({
@AssociationOverride(name = JpaMany3ManyEntity.ASOCIATION_OVERRIDE_ONE,
joinColumns =
@JoinColumn(name = "changeMeWhenExtends")),
@AssociationOverride(name = JpaMany3ManyEntity.ASOCIATION_OVERRIDE_TWO,
joinColumns =
@JoinColumn(name = "changeMeWhenExtends")),
@AssociationOverride(name = JpaMany3ManyEntity.ASOCIATION_OVERRIDE_THREE,
joinColumns =
@JoinColumn(name = "changeMeWhenExtends"))})
public abstract class JpaMany3ManyEntity<A extends JpaBaseEntity, B extends JpaBaseEntity, C extends JpaBaseEntity> extends JpaBaseEntity {

public static final String ID_ATTNAME = "id";

public static final String ASOCIATION_OVERRIDE_ONE = JpaMany3ManyEntity.ID_ATTNAME + "." + JpaMany3ManyId.ID_ONE_ATTNAME;

public static final String ASOCIATION_OVERRIDE_TWO = JpaMany3ManyEntity.ID_ATTNAME + "." + JpaMany3ManyId.ID_TWO_ATTNAME;

public static final String ASOCIATION_OVERRIDE_THREE = JpaMany3ManyEntity.ID_ATTNAME + "." + JpaMany3ManyId.ID_THREE_ATTNAME; 

 ...
 }

【问题讨论】:

  • 您介意添加JpaMany3ManyEntity 的定义以确保完整性吗?
  • 没问题,但我很确定这不是问题所在。无论如何,问题已经被编辑了。

标签: java hibernate inheritance jpa


【解决方案1】:

这是一个对我有用的解决方法:代理实体。

即使拥有实体的父代理 (jpa.inheritance.issue.Element_$$_javassist_1),如果您对它进行代理,您将获得真正的实体(子实体)。

假设您想从实体 A 循环您的(子)元素集合并对它们执行一些操作。

类似:

public void  loopDeproxyElements(List<Element> yourElementsCollection){
  for(Element p : yourElementsCollection){
      if(p instanceof HibernateProxy){
        Element child =   (Element) ((HibernateProxy) p).getHibernateLazyInitializer()
                    .getImplementation();

        if (child instanceof ElementTypeOne){

         //You can cast your object or do whatever you want, knowing for sure that's a child element)

          ElementTypeOne myRealElement =  (ElementTypeOne) child;
          ...
          } else {
           //It should be ElementTypeTwo (if u never create parent entities)
           ...
      }
    }        
  }
)

它总是会得到我所期望的子元素。

【讨论】:

  • 您能否添加示例代码以说明您如何实际使用它?
  • 这实际上是一个示例,基于问题中定义的类,您从惰性集合中获取值,但对值进行代理,始终获取实体的真实类型。意思是 p 是集合中的一个元素。
  • “p”从何而来?
  • 我已经编辑了答案。我希望它能帮助您了解 deproxy 的实际工作原理
  • 您的回答容易受到ClassCastException 的攻击。在铸造之前检查p instanceof HibernateProxy。有时,当实体在缓存中找到时,它可能已经被加载。
【解决方案2】:

尝试使用hibernate.default_batch_fetch_size 属性进行实验。默认情况下,它设置为 1。这将只加载集合中的第一个实体。将其增加到 ~ 集合的大小可能会有所帮助。

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2018-09-15
    • 1970-01-01
    • 2011-07-14
    • 2011-11-27
    • 1970-01-01
    相关资源
    最近更新 更多