0

次のテーブル構造を考慮してください

country
=======
id
code
name_id

label
======
id
code
label_value_id

translations
=============
id
ref_id
language
value

country:name_id次に、マップおよびlabel:label_value_id翻訳へのJPAマッピングを見つける必要がありますref_id。私はこの状況を説明するために正しい英語の用語の後にグーグルしてきましたが、まともなヒットで空っぽになりつつあります。したがって、データベースはこの例のようにレコードを保持できます

country
id: 1, code: BE, name_id: 30

label
id: 1, code: LABELA, label_value_id: 31

translations
id: 1, ref_id: 30, language: EN, value: BELGIUM
id: 2, ref_id: 30, language: NL, value: BELGIE
id: 3, ref_id: 31, language: EN, value: ALPHA_A
id: 4, ref_id: 31, language: NL, value: ALFA_A

Javaには3つのクラスがあります

国、ラベル、翻訳ここで@OneToMany、国とラベルから翻訳への関係を作成します。どちらもマップする必要がありますが、これを実現するためのコードref_idの記述方法がわかりません。@OneToMany正しい方向へのヒント、解決策またはマニュアルをいただければ幸いです

=====

2013年3月23日更新

Joopが述べたように、弁別器を使用することは解決策ですが、箱から出してすぐには機能しませんでした。休止状態のアノテーション@DiscriminatorOptions(force = true)を使用することを余儀なくされました。追加しない場合、hibernateは必要なコレクションをフェッチするときにSQLクエリのDiscriminatorを完全に無視します。

@Entity
@Table(name = "testtranslations")
@Inheritance(strategy = InheritanceType.SINGLE_TABLE)
@DiscriminatorColumn(name = "DTYPE", discriminatorType = DiscriminatorType.STRING)
@DiscriminatorOptions(force=true)
public class TestTranslation extends DomainObject {

    /**
     * 
     */
    private static final long serialVersionUID = -6211853644196769521L;

    private long id;
    private String language;
    private String value;

    @Id
    @GeneratedValue(strategy=GenerationType.AUTO, generator="testtranslations_seq_gen") 
    @SequenceGenerator(name="testtranslations_seq_gen", sequenceName="TESTTRANSLATIONS_SEQ") 
    @Column(name="testtranslation_id")
    public long getId() {
        return id;
    }

    @Column(name="language", length=3, nullable=false)
    public String getLanguage() {
        return language;
    }

    @Column(name="value")
    public String getValue() {
        return value;
    }

    @Override
    public void setId(long id) {
        this.id = id;
    }

    public void setLanguage(String language) {
        this.language = language;
    }

    public void setValue(String value) {
        this.value = value;
    }
}


@Entity
@Table(name = "testcountries")
public class TestCountry extends DomainObject {

    /**
     * 
     */
    private static final long serialVersionUID = -9207081478447113501L;

    private long id;
    private String code;
    private List<NameTranslation> name;
    private List<DescriptionTranslation> description;

    @Id
    @GeneratedValue(strategy=GenerationType.AUTO, generator="testcountries_seq_gen") 
    @SequenceGenerator(name="testcountries_seq_gen", sequenceName="TESTCOUNTRIES_SEQ") 
    @Column(name="country_id")
    public long getId() {
        return id;
    }

    @Column(name="iso_code", length=3, nullable=false)
    public String getCode() {
        return code;
    }

    @OneToMany(mappedBy="refId")
    public List<NameTranslation> getName() {
        return name;
    }

    @OneToMany(mappedBy="refId")
    public List<DescriptionTranslation> getDescription() {
        return description;
    }

    @Override
    public void setId(long id) {
        this.id = id;
    }

    public void setCode(String code) {
        this.code = code;
    }

    public void setName(List<NameTranslation> name) {
        this.name = name;
    }

    public void setDescription(List<DescriptionTranslation> description) {
        this.description = description;
    }
}

@Entity
@DiscriminatorValue("NAMETRANSLATION")
public class NameTranslation extends TestTranslation {

    /**
     * 
     */
    private static final long serialVersionUID = 7197732491071768673L;

    private TestCountry refId;

    @ManyToOne(fetch=FetchType.LAZY)
    @JoinColumn(name = "refId", nullable=false)
    public TestCountry getRefId() {
        return refId;
    }

    public void setRefId(TestCountry refId) {
        this.refId = refId;
    }   
}

@Entity
@DiscriminatorValue("DESCTRANSLATION")
public class DescriptionTranslation extends TestTranslation {

    /**
     * 
     */
    private static final long serialVersionUID = -4128287237786410515L;

    private TestCountry refId;

    @ManyToOne(fetch=FetchType.LAZY)
    @JoinColumn(name = "refId", nullable=false)
    public TestCountry getRefId() {
        return refId;
    }

    public void setRefId(TestCountry refId) {
        this.refId = refId;
    }   
}

必要な休止状態のマッピングを作成し、次のデータを使用してIDでTestCountryをロードするDbUnitテストを作成しました

<TESTCOUNTRIES COUNTRY_ID="1" VERSION="0" ISO_CODE="BE" />  
<TESTTRANSLATIONS TESTTRANSLATION_ID="1" VERSION="0" LANGUAGE="EN" VALUE="Belgium" REFID="1" DTYPE="NAMETRANSLATION" />
<TESTTRANSLATIONS TESTTRANSLATION_ID="2" VERSION="0" LANGUAGE="NL" VALUE="Belgie" REFID="1" DTYPE="NAMETRANSLATION" />
<TESTTRANSLATIONS TESTTRANSLATION_ID="3" VERSION="0" LANGUAGE="EN" VALUE="BelgiumDesc" REFID="1" DTYPE="DESCTRANSLATION" />
<TESTTRANSLATIONS TESTTRANSLATION_ID="4" VERSION="0" LANGUAGE="NL" VALUE="BelgieDesc" REFID="1" DTYPE="DESCTRANSLATION" />

これが将来他の人々に役立つことを願っています。JPAソリューションがないのは悲しいことであり、休止状態の注釈を使用することを余儀なくされました。

4

1 に答える 1

1

テーブル 1 またはテーブル 2 のいずれかの外部キーがあるため、少し難しいです。

テーブル継承のために、JPAには識別子の概念存在します。次に、共通の抽象テーブルから 2 つのテーブルを派生させ、異なる識別子フィールドを持ちます。一。ただし、これは少し異なる動作をします。

PS 検索キーの discriminatorValue と discriminatorColumn を使用して、より良い例を探します。

于 2013-03-21T15:11:40.570 に答える