2017-01-24 15 views
0

DiscriminatorColumnの使用中にデータベースからテーブルを読み込む際に問題があります。私のスーパークラス(ObjectA)を拡張するエンティティが存在する限り、データロードに問題はありません。 discriminatorカラムに新しい値が現れ、テーブル拡張がない場合に問題が発生しました。注釈(ObjectTypes.GENERIC_OBJECT)で指定されているものとは異なる識別子の値でObjctAをロードしようとしています。私はIngeritanceType.JOINEDを使用しているので、これらのテーブルを結合するidを持つ子テーブルがあるはずですが、そのようなテーブルはありません。私もそれを追加することはできません(要件)。Hibernate継承の問題


私がしようとしているのは、ObjectAクラスを、注釈で指定されたものとは異なる識別子の値でロードすることです。私は値がただ1つの値を受け入れることを知っています。また、ObjectAを拡張した空のクラスを作成して、dicriminatorの値を目的の値に設定しようとしました。失敗。 Hibernateはエンティティとしてそれを扱い、それを見つけることができません。

このシナリオの回避策はありますか?

@MappedSuperclass 
public abstract class AbstractEntity implements Serializable 
{... 

@Entity 
@Table(name = "objecta") 
@Inheritance(strategy = InheritanceType.JOINED) 
@DiscriminatorColumn(name = "objectclass", discriminatorType = DiscriminatorType.INTEGER) 
@DiscriminatorValue(ObjectTypes.GENERIC_OBJECT) 
@NamedQueries({...}) 
public class ObjectA extends AbstractEntity { 
... 

@Entity 
@Table(name = "objectb") 
@DiscriminatorValue(ObjectTypes.OBJECT_B) 
@NamedQueries({...}) 
public class ObjectB extends ObjectA{ 

public class ObjectTypes{ 
    public static final String GENERIC_OBJECT = "0"; 
    public static final String OBJECT_B= "1"; 
    public static final String OBJECT_C = "2";//no extension table for that object type, 
//it is the same as GENERI_OBJECT (no additional properties) 
    public static final String OBJECT_C = "3"; 
...} 

(編集)発行:


私のエンティティは、このようになります

私はネイティブのクエリを実行することにより、ObjectAにオブジェクトをロードすることはできません(ただし、ときObjectAに、私はそれを読み込むことができます子オブジェクト拡張としてロードされます)。

@NamedQueries({ @NamedQuery(name = ObjectA.FIND_BY_ID, query = "SELECT t FROM ObjectA t WHERE t.id = :id"), 
     @NamedQuery(name = ObjectA.FIND_ALL, query = "SELECT t FROM ObjectA t"), 
     @NamedQuery(name = ObjectA.TOTAL_RESULT, query = "SELECT count(t) from ObjectA t"), 
     @NamedQuery(name = ObjectA.FIND_BY_OBJECT_TYPE, query = "SELECT t FROM ObjectA t WHERE t.objecttype = :objectType") }) 
public class ObjectA extends AbstractEntity { 

    private static final long serialVersionUID = 8831394270398606926L; 
    public static final String FIND_BY_ID = "ObjectA.find"; 
    public static final String FIND_ALL = "ObjectA.findAll"; 
... 

//loading part 
    public List<ObjectA> findByObjectType(ObjectTypes objectType) { 
     return em.createNamedQuery(ObjectA.FIND_BY_OBJECT_TYPE, ObjectA.class).setParameter("objectType", objectType).getResultList(); 
    } 

スタックトレース:InheritanceType.JOINEDを使用している場合

Caused by: javax.persistence.PersistenceException: org.hibernate.type.SerializationException: could not deserialize 
    at org.hibernate.jpa.spi.AbstractEntityManagerImpl.convert(AbstractEntityManagerImpl.java:1692) 
    at org.hibernate.jpa.spi.AbstractEntityManagerImpl.convert(AbstractEntityManagerImpl.java:1602) 
    at org.hibernate.jpa.internal.QueryImpl.getResultList(QueryImpl.java:492) 
    at com.mypackage.repository.ObjectARepository.findAll(ObjectARepository.java:38) 
    at com.mypackage.database.DatabaseManager.init(DatabaseManager.java:77) 
    ... 240 more 
Caused by: org.hibernate.type.SerializationException: could not deserialize 
    at org.hibernate.internal.util.SerializationHelper.doDeserialize(SerializationHelper.java:243) 
    at org.hibernate.internal.util.SerializationHelper.deserialize(SerializationHelper.java:287) 
    at org.hibernate.type.descriptor.java.SerializableTypeDescriptor.fromBytes(SerializableTypeDescriptor.java:138) 
    at org.hibernate.type.descriptor.java.SerializableTypeDescriptor.wrap(SerializableTypeDescriptor.java:113) 
    at org.hibernate.type.descriptor.java.SerializableTypeDescriptor.wrap(SerializableTypeDescriptor.java:27) 
    at org.hibernate.type.descriptor.sql.VarbinaryTypeDescriptor$2.doExtract(VarbinaryTypeDescriptor.java:60) 
    at org.hibernate.type.descriptor.sql.BasicExtractor.extract(BasicExtractor.java:47) 
    at org.hibernate.type.descriptor.converter.AttributeConverterSqlTypeDescriptorAdapter$2.extract(AttributeConverterSqlTypeDescriptorAdapter.java:121) 
    at org.hibernate.type.AbstractStandardBasicType.nullSafeGet(AbstractStandardBasicType.java:238) 
    at org.hibernate.type.AbstractStandardBasicType.nullSafeGet(AbstractStandardBasicType.java:234) 
    at org.hibernate.type.AbstractStandardBasicType.nullSafeGet(AbstractStandardBasicType.java:224) 
    at org.hibernate.type.AbstractStandardBasicType.hydrate(AbstractStandardBasicType.java:300) 
    at org.hibernate.persister.entity.AbstractEntityPersister.hydrate(AbstractEntityPersister.java:2738) 
    at org.hibernate.loader.Loader.loadFromResultSet(Loader.java:1729) 
    at org.hibernate.loader.Loader.instanceNotYetLoaded(Loader.java:1655) 
    at org.hibernate.loader.Loader.getRow(Loader.java:1544) 
    at org.hibernate.loader.Loader.getRowFromResultSet(Loader.java:727) 
    at org.hibernate.loader.Loader.processResultSet(Loader.java:972) 
    at org.hibernate.loader.Loader.doQuery(Loader.java:930) 
    at org.hibernate.loader.Loader.doQueryAndInitializeNonLazyCollections(Loader.java:336) 
    at org.hibernate.loader.Loader.doList(Loader.java:2617) 
    at org.hibernate.loader.Loader.doList(Loader.java:2600) 
    at org.hibernate.loader.Loader.listIgnoreQueryCache(Loader.java:2429) 
    at org.hibernate.loader.Loader.list(Loader.java:2424) 
    at org.hibernate.loader.hql.QueryLoader.list(QueryLoader.java:501) 
    at org.hibernate.hql.internal.ast.QueryTranslatorImpl.list(QueryTranslatorImpl.java:371) 
    at org.hibernate.engine.query.spi.HQLQueryPlan.performList(HQLQueryPlan.java:216) 
    at org.hibernate.internal.SessionImpl.list(SessionImpl.java:1326) 
    at org.hibernate.internal.QueryImpl.list(QueryImpl.java:87) 
    at org.hibernate.jpa.internal.QueryImpl.list(QueryImpl.java:606) 
    at org.hibernate.jpa.internal.QueryImpl.getResultList(QueryImpl.java:483) 
    ... 242 more 
Caused by: java.io.StreamCorruptedException: invalid stream header: 7B227363 
    at java.io.ObjectInputStream.readStreamHeader(ObjectInputStream.java:806) 
    at java.io.ObjectInputStream.<init>(ObjectInputStream.java:299) 
    at org.hibernate.internal.util.SerializationHelper$CustomObjectInputStream.<init>(SerializationHelper.java:309) 
    at org.hibernate.internal.util.SerializationHelper$CustomObjectInputStream.<init>(SerializationHelper.java:299) 
    at org.hibernate.internal.util.SerializationHelper.doDeserialize(SerializationHelper.java:218) 
    ... 272 more 
+0

objectTypeはどのようにマップされていますか?問題は、結果の中の何かをデシリアライズすることに関連しているようです。関連するマッピング(変数)を表示する – Tobb

+0

これは 'javax.persistence'コンバータによってマップされます。しかし、それは問題ではありません。異なるObjectAプロパティで無効な 'jasonb'変換によって例外がスローされます。 Tobbのお手伝いをありがとう。下のあなたの答えは私を助けました。 – AndreyB

答えて

0

(これはSINGLE_TABLE継承のためのものです)DiscriminatorColumnための必要はありません。

InheritanceType.JOINEDについて、あなたはあなたのサブクラスで、以下が必要になります。

@PrimaryKeyJoinColumn(name = "parent_id", referencedColumnName = "id") 

あなたのサブクラスのテーブルには、両方のテーブルのprimary keyである列parent_id、および列のforeign keyが必要になりますスーパークラスに対応する表のid(主キー)。

+0

いいえ、私はDiscriminatorColumnを削除し、 '@ PrimaryKey..'を追加しました。しかし、私の大きな問題は残っています。私はObjectAオブジェクトのいくつかを取得するためにネイティブクエリを実行しようとしていますが、私は逆直列化例外「無効なストリームヘッダ...」を取得します。あなたはその理由を知っていますか? ObjectAはdbから親としてロードできますが、単独でロードすることはできません。 – AndreyB

+0

質問にクエリと例外を追加します。 – Tobb