Класс Hibernate @Embeddable, который расширяет другой класс @Embeddable, Свойства не найдены для сопоставления @OneToMany

Мы переводим старую конфигурацию на основе xml в конфигурацию на основе аннотаций

Ситуация

Существует класс, который аннотируется как @Embeddable(ParentPk.java), другой класс расширяет этот класс, который является @Embeddable(ChildPk.java), этот ChildPk.java используется как составной первичный ключ в SomeOwnerClass.java, который имеет внешнее отношение с другим классом SomeChildTable.java и имеет тенденцию использовать свойства col1 и col2, которые доступны в родительском классе ChildPk.java, но когда запрос выполняется, hibernate не находит col1 и col2, если я скопирую col1 и col2 в ChildPk.java из родительского класса, все прекрасно работает.

Ниже приведен фрагмент кода SomeOwnerClass.java, который ссылается на col1 и col2

>  @OneToMany(orphanRemoval = true, fetch=FetchType.EAGER)
>        @Cascade(value = {CascadeType.ALL,CascadeType.SAVE_UPDATE})
>        @Fetch(FetchMode.JOIN)
>        @JoinColumns({
>        @JoinColumn(name="COL_1",insertable=false,updatable=false,referencedColumnName="COL_1"),
>        @JoinColumn(name="COL_2",insertable=false,updatable=false,referencedColumnName="COL_2"),
>        @JoinColumn(name="COL_3",insertable=false,updatable=false,referencedColumnName="COL_3"),
>     })    private Set<SomeChildTable> collection = new
> HashSet<SomeChildTable>();

Любое решение для этой ситуации?

ParentPk.java

 @Embeddable
    public class ParentPk implements Serializable  {

        @Column(name="COL_1")
        private String col1;

        @Column(ame="COL_2")
        private String col2;

    }

ChildPk.java

@Embeddable
public class ChildPk extends ParentPk implements Serializable  {

    @Column(name="COL_3")
    private String col3;
}

SomeOwnerClass.java

@Entity
@Table(name="FOO")
public class SomeOwnerClass implements Serializable  {

    @EmbeddedId
       @AttributeOverrides({@AttributeOverride(name="col1", [email protected](name="COL_1",length=38))})
        private ChildPk childPk = new ChildPk();

         @OneToMany(orphanRemoval = true, fetch=FetchType.EAGER)
         @Cascade(value = {CascadeType.ALL,CascadeType.SAVE_UPDATE})
         @Fetch(FetchMode.JOIN)
         @JoinColumns({
         @JoinColumn(name="COL_1",insertable=false,updatable=false,referencedColumnName="COL_1"),
         @JoinColumn(name="COL_2",insertable=false,updatable=false,referencedColumnName="COL_2"),
         @JoinColumn(name="COL_3",insertable=false,updatable=false,referencedColumnName="COL_3"),
      })
    private Set<SomeChildTable> collection = new HashSet<SomeChildTable>();
}

Exception

org.hibernate.MappingException: Unable to find column with logical name: COL_1 in SomeOwnerClass
    at org.hibernate.cfg.Ejb3JoinColumn.checkReferencedColumnsType(Ejb3JoinColumn.java:587)
    at org.hibernate.cfg.BinderHelper.createSyntheticPropertyReference(BinderHelper.java:258)
    at org.hibernate.cfg.annotations.CollectionBinder.bindCollectionSecondPass(CollectionBinder.java:1451)
    at org.hibernate.cfg.annotations.CollectionBinder.bindOneToManySecondPass(CollectionBinder.java:864)
    at org.hibernate.cfg.annotations.CollectionBinder.bindStarToManySecondPass(CollectionBinder.java:779)
    at org.hibernate.cfg.annotations.CollectionBinder$1.secondPass(CollectionBinder.java:728)
    at org.hibernate.cfg.CollectionSecondPass.doSecondPass(CollectionSecondPass.java:70)
    at org.hibernate.cfg.Configuration.originalSecondPassCompile(Configuration.java:1695)
    at org.hibernate.cfg.Configuration.secondPassCompile(Configuration.java:1424)
    at org.hibernate.cfg.Configuration.buildSessionFactory(Configuration.java:1844)
    at com.foo.mypackage.ParentHibernateUtil.initiateSessionFactory(ParentHibernateUtil.java:112)
    at com.foo.mypackage.ParentHibernateUtil.getSessionFactory(ParentHibernateUtil.java:131)
    at com.foo.mypackage.GenericHibernateDAO.getSession(GenericHibernateDAO.java:36)
    at com.foo.mypackage.GenericHibernateDAO.beginTransaction(GenericHibernateDAO.java:63)
    at com.foo.mypackage.MarsTest.main(MyTest.java:22)

Дополнительная информация Я попытался сопоставить конфигурацию на основе xml для конфигурации, основанной на аннотации, ниже - старая конфигурация на основе xml, которая отлично работает с точки зрения наследования.

ChildPk.java преобразован в @Embedable, который расширяет ParentPk.java

<class name="SomeOwnerClassDetail" table="FOO_DETAIL">

        <composite-id class="ChildPk"
            name="childPk">
            <key-property name="col1" column="COL_1"  type="java.lang.Long"/>
            <key-property name="col2" column="COL_2" length="32"/>
            <key-property name="col3" column="COL_3" length="3"/>
        </composite-id>
    </class>

в приведенных выше сопоставлениях col1 и col2 наследуются от ParentPk.java, которые доступны, если ChildPk используется как внешний ключ в SomeOwnerClass.java в сопоставлении xml, но не в аннотированном сопоставлении.

Я не могу изменить структуру своего класса, поскольку это устаревшее приложение.

SomeOwnerClass.java

<set name="someDetailKey" cascade="all,delete-orphan,save-update" lazy="false" fetch="join"  > 
        <key foreign-key="FOO_Foreign_Key" >
            <column name="COL_1"/>
            <column name="COL_2"/>
            <column name="COL_3"/>
        </key>
        <one-to-many class="ChildPk" />
    </set>

Ответ 1

Hibernate не поддерживает наследование компонента @Embeddable:

Встраиваемый объект не может быть непосредственно сохранен или запрошен, он может быть сохранен или запрошен только в контексте его родителя. Вложенный объект не имеет идентификатора или таблицы. Спецификация JPA не поддерживает встраиваемые объекты, имеющие наследование, хотя некоторые поставщики JPA могут это допускать.

- Java_Persistence/Embeddables.

Спецификация JPA не поддерживает вложенные объекты с наследованием
Поэтому вы не можете использовать Inheritance в @Embeddable.

Вы можете попытаться сделать это, указав родительский id как свойство дочернего идентификатора.

Ответ 2

Вы можете реализовать наследование между классами @Embeddable. Вам просто нужно аннотировать родительский класс с помощью @MappedSuperclass.

Итак, например:

@Embeddable
@MappedSuperclass
public class Parent {
    @Basic
    private String parentProperty;

    // ... getters/setters
}

@Embeddable
public class Child extends Parent {
    @Basic
    private String childProperty;

    // ... getters/setters
}

Таким образом, Hibernate (проверенный с 5.x) правильно отобразит как parentProperty, так и childProperty в классе Child.