Связь JPA «один ко многим» с объектом с @Inheritance(strategy=InheritanceType.JOINED)

Привет всем.

Я пытаюсь сопоставить ассоциацию «один ко многим» с сущностью, которая сопоставлена ​​с @Inheritance(strategy=InheritanceType.JOINED)

....
@ManyToMany
@JoinTable(name = "S_MC_CC_CONTRATTIRAPPORTI",
    joinColumns = @JoinColumn(name = "COD_MULTICHANNELID"),
    inverseJoinColumns = @JoinColumn(name = "COD_RAPPORTO")
)   
private Collection<Rapporto> rapporti;
...

@Entity
@Table(name = "S_MC_CC_RAPPORTI")
@Inheritance(strategy=InheritanceType.JOINED)
@DiscriminatorColumn(name="COD_TIPORAPPORTO")
public abstract class Rapporto implements Serializable {

Проблема в том, что я получаю это исключение Описание исключения: Отсутствует класс для значения поля индикатора [1] типа [класс java.lang.String].

Я думаю, что JPA пытается определить DiscriminatorValue для класса. Я хотел бы загрузить все объекты, расширяющие суперкласс.

Является ли это возможным?

Дополнительная информация: это владелец ассоциации

@Entity  
@Table(name = "S_MC_CC_CONTRATTI")
public class Contratto implements Serializable {

    @Id
    @Column(name = "COD_MULTICHANNELID")
    private String multichannelId;

    @ManyToMany()
    @JoinTable(name = "S_MC_CC_CONTRATTIRAPPORTI", joinColumns = @JoinColumn(name = "COD_MULTICHANNELID"), inverseJoinColumns = @JoinColumn(name = "COD_RAPPORTO")) 
    private Collection<Rapporto> rapporti;

Это абстрактный класс ассоциации «многие ко многим».

@Entity
@Table(name = "S_MC_CC_RAPPORTI")
@Inheritance(strategy=InheritanceType.JOINED)
@DiscriminatorColumn(name="COD_TIPORAPPORTO")
public abstract class Rapporto implements Serializable {

    private static final long serialVersionUID = -5567166522882040440L;

    @Id
    @Column(name = "COD_RAPPORTO")  
    protected Long codiceRapporto;

А это два подкласса:

@Entity
@Table(name="S_MC_CC_CCCLIENTI")
@DiscriminatorValue("1 ")
public class ContoCorrente extends Rapporto {
    private static final long serialVersionUID = -3380622649760983262L;

    @Column(name = "DESC_DIVISA")
    private String divisa;

@Entity
@Table(name = "S_MC_CC_RAPPORTI")
@DiscriminatorValue("6 ")
public class NumeroPortafoglioEstero extends Rapporto {

    private static final long serialVersionUID = 6869743707936663970L;

    /**
     * @param filiale
     * @param categoria
     * @param conto
     */
    public NumeroPortafoglioEstero(Filiale filiale, String categoria, String conto) {
        super(filiale, categoria, conto);
    }

Это исключение:

Local Exception Stack: 
Exception [EclipseLink-43] (Eclipse Persistence Services - 2.1.2.v20101206-r8635): org.eclipse.persistence.exceptions.DescriptorException
Exception Description: Missing class for indicator field value [1] of type [class java.lang.String].
Descriptor: RelationalDescriptor(it.alten.intesasanpaolo.contratto.domain.core.rapporto.Rapporto --> [DatabaseTable(S_MC_CC_RAPPORTI)])
    at org.eclipse.persistence.exceptions.DescriptorException.missingClassForIndicatorFieldValue(DescriptorException.java:921)
    at org.eclipse.persistence.descriptors.InheritancePolicy.classFromValue(InheritancePolicy.java:355)
    at org.eclipse.persistence.descriptors.InheritancePolicy.classFromRow(InheritancePolicy.java:342)
    at org.eclipse.persistence.descriptors.InheritancePolicy.selectAllRowUsingDefaultMultipleTableSubclassRead(InheritancePolicy.java:1303)
    at org.eclipse.persistence.descriptors.InheritancePolicy.selectAllRowUsingMultipleTableSubclassRead(InheritancePolicy.java:1402)
    at org.eclipse.persistence.internal.queries.ExpressionQueryMechanism.selectAllRows(ExpressionQueryMechanism.java:2493)
    at org.eclipse.persistence.queries.ReadAllQuery.executeObjectLevelReadQuery(ReadAllQuery.java:407)
    at org.eclipse.persistence.queries.ObjectLevelReadQuery.executeDatabaseQuery(ObjectLevelReadQuery.java:1076)
    at org.eclipse.persistence.queries.DatabaseQuery.execute(DatabaseQuery.java:740)
    at org.eclipse.persistence.queries.ObjectLevelReadQuery.execute(ObjectLevelReadQuery.java:1036)
    at org.eclipse.persistence.queries.ReadAllQuery.execute(ReadAllQuery.java:380)
    at org.eclipse.persistence.internal.sessions.AbstractSession.internalExecuteQuery(AbstractSession.java:2392)
    at org.eclipse.persistence.internal.sessions.AbstractSession.executeQuery(AbstractSession.java:1291)
    at org.eclipse.persistence.internal.sessions.AbstractSession.executeQuery(AbstractSession.java:1273)
    at org.eclipse.persistence.internal.indirection.QueryBasedValueHolder.instantiate(QueryBasedValueHolder.java:96)
    at org.eclipse.persistence.internal.indirection.QueryBasedValueHolder.instantiate(QueryBasedValueHolder.java:86)
    at org.eclipse.persistence.internal.indirection.DatabaseValueHolder.getValue(DatabaseValueHolder.java:83)
    at org.eclipse.persistence.internal.indirection.UnitOfWorkValueHolder.instantiateImpl(UnitOfWorkValueHolder.java:160)
    at org.eclipse.persistence.internal.indirection.UnitOfWorkValueHolder.instantiate(UnitOfWorkValueHolder.java:220)
    at org.eclipse.persistence.internal.indirection.DatabaseValueHolder.getValue(DatabaseValueHolder.java:83)
    at org.eclipse.persistence.indirection.IndirectList.buildDelegate(IndirectList.java:237)
    at org.eclipse.persistence.indirection.IndirectList.getDelegate(IndirectList.java:398)
    at org.eclipse.persistence.indirection.IndirectList.toArray(IndirectList.java:741)
    at it.alten.intesasanpaolo.contratto.domain.core.anagrafica.Contratto.toString(Contratto.java:262)
    at java.lang.String.valueOf(String.java:2827)
    at java.io.PrintStream.println(PrintStream.java:771)
    at it.alten.intesasanpaolo.contratto.dao.core.anagrafica.TestContrattoDaoJpaImpl.testGetItemByID(TestContrattoDaoJpaImpl.java:99)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:39)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25)
    at java.lang.reflect.Method.invoke(Method.java:597)
    at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:44)
    at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:15)
    at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:41)
    at org.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:20)
    at org.junit.internal.runners.statements.RunBefores.evaluate(RunBefores.java:28)
    at org.junit.runners.BlockJUnit4ClassRunner.runNotIgnored(BlockJUnit4ClassRunner.java:79)
    at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:71)
    at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:49)
    at org.junit.runners.ParentRunner$3.run(ParentRunner.java:193)
    at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:52)
    at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:191)
    at org.junit.runners.ParentRunner.access$000(ParentRunner.java:42)
    at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:184)
    at org.junit.runners.ParentRunner.run(ParentRunner.java:236)
    at org.eclipse.jdt.internal.junit4.runner.JUnit4TestReference.run(JUnit4TestReference.java:49)
    at org.eclipse.jdt.internal.junit.runner.TestExecution.run(TestExecution.java:38)
    at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:467)
    at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:683)
    at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.run(RemoteTestRunner.java:390)
    at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.main(RemoteTestRunner.java:197)

С уважением Массимо


person Massimo Ugues    schedule 14.01.2011    source источник
comment
Какой поставщик JPA вы используете?   -  person axtavt    schedule 14.01.2011


Ответы (3)


Каждая сущность в иерархии наследования должна быть классом с аннотацией @Entity и @DiscriminatorValue.

person Bozho    schedule 14.01.2011
comment
Хорошо, и я сделал это. Проблема заключается в ассоциации с суперклассом, который является абстрактным и не имеет DiscriminatorValue. Imagine Person --› Animals, где Animal — это абстрактный суперкласс, расширенный классами Cat и Dog. Cat и Dog имеют DiscriminatorValue, а Animal — нет. Теперь я хотел бы ассоциироваться от Человека к Животному и иметь всех кошек и собак. Мне нужно что-то вроде: targetEntity = {Cat.class, Dog.class}, но это не поддерживается. Любая идея? С уважением - person Massimo Ugues; 14.01.2011
comment
Затем @m.ugues предоставит дополнительную информацию - покажите свои дочерние классы и точное исключение с помощью трассировки стека. - person Bozho; 14.01.2011

Согласно документации EclipseLink (см.), это исключение возникает, когда класс с определенным значением дискриминатора не может быть найден в иерархии наследования.

Обратите внимание, что в сообщении об ошибке значение дискриминатора указано как 1, тогда как в @DiscriminatorValue оно указано как "1 ". Я думаю, что это дополнительное пространство вызывает проблему - либо это пространство удаляется при загрузке значения из базы данных, либо старые значения в базе данных имеют дискриминаторы без пробелов.

person axtavt    schedule 14.01.2011

Я внес изменения в @DiscriminatorValue: теперь суперкласс определяет: @DiscriminatorColumn(name="COD_TIPORAPPORTO", discriminatorType=DiscriminatorType.STRING, length=2)
и родственные подклассы:
@DiscriminatorValue("1").
Теперь я больше не получаю исключений, но запросы, сгенерированные для получения данных, не работают: вот пример того, что генерируется:

[EL Fine]: 2011-01-14 19:29:53.312--ServerSession(5230193)--Connection(5612344)--Thread(Thread[main,5,main])--SELECT t0.COD_RAPPORTO, t0.COD_TIPORAPPORTO, t0.COD_MATRICOLAGESTORE, t0.FLAG_RENDICONTAZIONE, t0.COD_CATEGORIA, t0.COD_STATORAPPORTO, t0.FLAG_SB_ATTIVO, t0.COD_GESTORE, t0.TIMESTAMP_AGGIORNAMENTO, t0.DESC_ATTRIBUTO, t0.NUM_RAPPORTO, t0.COD_NSG, t0.COD_ABI, t0.DEN_FILIALE, t1.COD_RAPPORTO, t1.COD_CIN, t1.DESC_DIVISA, t1.COD_CAB, t1.COD_IBAN, t1.COD_CHD, t1.COD_NAZIONE, t1.COD_SERVIZIO FROM S_MC_CC_RAPPORTI t0, S_MC_CC_CONTRATTIRAPPORTI t2, S_MC_CC_CCCLIENTI t1 WHERE (((t2.COD_MULTICHANNELID = ?) AND (t0.COD_RAPPORTO = t2.COD_RAPPORTO)) AND ((t1.COD_RAPPORTO = t0.COD_RAPPORTO) AND (t0.COD_TIPORAPPORTO = ?)))

привязать => [01025640171576493, 1]

Фактически запрос, отправленный через Toad, работает нормально. Кажется, что запросы на загрузку всех данных работают, но на самом деле я не вижу значения ассоциации.

С уважением ко всем...

person Massimo Ugues    schedule 14.01.2011