Наследование JPA и Hibernate Issue

У меня есть странная проблема с загрузкой некоторых объектов. Я использую 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, содержащая элементы Elements:

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

И если я получаю коллекцию, все работает нормально (я получу подклассы, как ожидалось).

Проблема возникает, когда у меня есть другой объект B с коллекцией JpaMany3ManyEntity (обратите внимание, что задействован один и тот же элемент сущности)

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

Если я зациклю три элемента ElementExampleCollection из класса B, прежде чем попытаться получить elementCollection из класса A, когда я загружаю объекты из elementCollection Я получаю только объекты суперкласса (Element) вместо детей.

Я предполагаю, что по какой-либо причине отношение "многие к многим" всегда получает объекты Element (отец) и сохраняет их в кэше спящего режима, но мне нужно избегать такого поведения.

Любые идеи или рабочий стол? Любая помощь будет действительно оценена.

Спасибо заранее.

EDIT: многие для многих классов:

@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; 

 ...
 }

Ответ 1

Здесь работает рабочий стол, который работает со мной: Deproxy сущности.

Даже имея родительский прокси объекта (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 a child element)

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

Он всегда будет получать дочерние элементы, как я ожидал.

Ответ 2

Попробуйте экспериментировать с свойством hibernate.default_batch_fetch_size. По умолчанию он устанавливается в 1. Это будет загружать только первый объект из вашей коллекции. Увеличение его до ~ размера коллекции может помочь.