3

私はHibernateとRESTeasyを使用していますが、ArtisteエンティティとOeuvreエンティティの間にOneToMany(ManyToOne)双方向の関係があるため、これらのエンティティとのサイクルを避けようとしています。

Oeuvre.java

import javax.persistence.*;
import javax.xml.bind.annotation.*;
import org.eclipse.persistence.oxm.annotations.XmlInverseReference;

@Entity
@Inheritance(strategy = InheritanceType.SINGLE_TABLE)
@XmlRootElement(name = "oeuvre")
public abstract class Oeuvre {

    @Id
    @GeneratedValue(strategy = GenerationType.AUTO)
    private int id;
    @Embedded
    private Dimension dimension;

    @XmlElement(defaultValue = "true")
    private boolean hasBeenReproduced;

    @ManyToOne(cascade = CascadeType.PERSIST, fetch = FetchType.LAZY)
    @JoinColumn(name = "artiste_id")
    @XmlIDREF
    private Artiste artiste;

    @XmlElement
    public int getId() {
        return id;
    }

    public void setId(int id) {
        this.id = id;
    }

    // @XmlTransient
    @XmlInverseReference(mappedBy = "oeuvres")
    public Artiste getArtiste() {
        return artiste;
    }

    public void setArtiste(Artiste artiste) {
        this.artiste = artiste;
        artiste.addOeuvre(this);
    }

}

Personne.java

import javax.persistence.*;
import javax.xml.bind.annotation.XmlID;

@Entity
@Inheritance(strategy = InheritanceType.TABLE_PER_CLASS)
public abstract class Personne {

    @Id
    @GeneratedValue(strategy = GenerationType.TABLE)
    @XmlID
    private int id;

}

Artiste.java

import java.util.*;
import javax.persistence.*;
import javax.xml.bind.annotation.*;

@Entity
@XmlRootElement(name = "artiste")
public class Artiste extends Personne {

    private String bibliographie;

    @OneToMany(mappedBy = "artiste", orphanRemoval = true, cascade = {
            CascadeType.PERSIST, CascadeType.REMOVE })
    private List<Oeuvre> oeuvres = new ArrayList<Oeuvre>();

    @XmlElement
    public List<Oeuvre> getOeuvres() {
        return oeuvres;
    }

    public void setOeuvres(List<Oeuvre> oeuvres) {
        this.oeuvres = oeuvres;
    }

}

だから私はMOXyを使うことにしました、

これが私のPOMです

<repository>
        <id>EclipseLink</id>
        <url>http://download.eclipse.org/rt/eclipselink/maven.repo</url>
</repository>
<dependency>
       <groupId>org.eclipse.persistence</groupId>
       <artifactId>org.eclipse.persistence.moxy </artifactId>
       <version>2.3.2</version>     
    </dependency>

nb:hibernateを使用しているのでorg.eclipse.persistence.moxy-2.3.2.jarのみが必要です(eclipseLinkは必要ありません)が、他に3つのjar(コアを含む)もあります

次に、jaxb.propertiesファイルをエンティティのパッケージに入れます。

javax.xml.bind.context.factory=org.eclipse.persistence.jaxb.JAXBContextFactory

そして、@ XmlInverseReference(mappedBy = "oeuvres")をgetArtiste()に追加しました。代わりにod @XmlTranscient ==>サイクルはもうありません(xmlTranscientのように)が、まだバックポインターがありません。

次に、@ XmlIDと@XmlIDREFを追加しました。アーティストのIDは、芸術作品のxml結果で表されますが、適切な値はありません(0ですが、他の値である必要があります)。

<Oeuvre>
  <hasBeenReproduced>false</hasBeenReproduced>
  <artiste>0</artiste>
  <year>2010</year>
  <id>2</id>
  <titre>La joconde</titre>
</Oeuvre>

私は何が間違っているのですか?事前にThx

編集 :

「Artiste」オブジェクトをマーシャルするときに、@XmlInverseReferenceを使用して次の出力があります。

<artiste>
  <id>1</id>
  <nom>a</nom>
  <prenom>b</prenom>
  <oeuvres>
     <hasBeenReproduced>false</hasBeenReproduced>
     <year>2010</year>
     <id>25</id>
     <titre>La joconde</titre>
  </oeuvres>
</artiste>    

あなたの例によると、これは正しい振る舞いです。したがって、私がよく理解している場合、「Oeuvre」出力(上記)にartisteidの参照を含めることはできません。芸術作品からアーティストを取得することはできません。私の場合、@ XmlIDを使用する必要はありませんか?

あなたの完全な答えのためのThxBlaiseDoughan、それは大いに感謝されます

4

1 に答える 1

6

注: 私はEclipseLink JAXB(MOXy)のリーダーであり、JAXB(JSR-222)エキスパートグループのメンバーです。

以下は、MOXyの@XmlInverseReference拡張機能を使用する方法の簡単な例です。

Foo

Fooこの例のルートオブジェクトです。

package forum13957068;

import java.util.List;

import javax.xml.bind.annotation.*;

@XmlRootElement
@XmlAccessorType(XmlAccessType.FIELD)
public class Foo {

    String fooProp;

    @XmlElement(name="bar")
    List<Bar> bars;

}

バー

Bar子オブジェクトです。@XmlInverseReference注釈を使用しfooて、親オブジェクトへの参照をフィールドに入力します。

package forum13957068;

import javax.xml.bind.annotation.*;
import org.eclipse.persistence.oxm.annotations.XmlInverseReference;

@XmlAccessorType(XmlAccessType.FIELD)
public class Bar {

    private String barProp;

    @XmlInverseReference(mappedBy="bars")
    Foo foo;

}

jaxb.properties

JAXBプロバイダーとしてMOXyを指定するにはjaxb.properties、次のエントリで呼び出されるファイルを含める必要があります。

javax.xml.bind.context.factory=org.eclipse.persistence.jaxb.JAXBContextFactory

input.xml

<?xml version="1.0" encoding="UTF-8"?>
<foo>
    <fooProp>A</fooProp>
    <bar>
        <barProp>B</barProp>
    </bar>
    <bar>
        <barProp>B</barProp>
    </bar>
</foo>

デモ

次のデモコードは、XMLのマーシャリングを解除し、バックポインターを確認してから、オブジェクトをマーシャリングしてXMLに戻します。

package forum13957068;

import java.io.File;
import javax.xml.bind.*;

public class Demo {

    public static void main(String[] args) throws Exception {
        JAXBContext jc = JAXBContext.newInstance(Foo.class);

        Unmarshaller unmarshaller = jc.createUnmarshaller();
        File xml = new File("src/forum13957068/input.xml");
        Foo foo = (Foo) unmarshaller.unmarshal(xml);

        for(Bar bar : foo.bars) {
            System.out.println(bar.foo == foo);
        }

        Marshaller marshaller = jc.createMarshaller();
        marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true);
        marshaller.marshal(foo, System.out);
    }

}

出力

以下は、デモコードを実行した結果です。バックポインタが正しく入力される方法をノード化します。

true
true
<?xml version="1.0" encoding="UTF-8"?>
<foo>
   <fooProp>A</fooProp>
   <bar>
      <barProp>B</barProp>
   </bar>
   <bar>
      <barProp>C</barProp>
   </bar>
</foo>

MOXyとMaven

ファイルで以下を使用してpom.xml、MOXyを取り込むことができます。

<dependencies>
    <dependency>
        <groupId>org.eclipse.persistence</groupId>
        <artifactId>org.eclipse.persistence.moxy</artifactId>
        <version>2.4.1</version>
    </dependency>
</dependencies>
<repositories>
    <repository>
        <id>EclipseLink Repo</id>
        <url>http://download.eclipse.org/rt/eclipselink/maven.repo</url>
    </repository>
</repositories>

MOXyを取り込むPOMファイルの例については、GitHubでホストしている例を確認してください。


アップデート

あなたの例によると、これは正しい振る舞いです。したがって、私がよく理解している場合、「Oeuvre」出力(上記)にartisteidの参照を含めることはできません。芸術作品からアーティストを取得することはできません。私の場合、@ XmlIDを使用する必要はありませんか?

以下に、の代わりに@XmlID/を使用する代替マッピングを示します。@XmlIDREF@XmlInverseReference

Foo

package forum13957068;

import java.util.List;
import javax.xml.bind.annotation.*;

@XmlRootElement
@XmlAccessorType(XmlAccessType.FIELD)
public class Foo {

    @XmlID
    String id;

    String fooProp;

    @XmlElement(name="bar")
    List<Bar> bars;

}

バー

package forum13957068;

import javax.xml.bind.annotation.*;

@XmlAccessorType(XmlAccessType.FIELD)
public class Bar {

    private String barProp;

    @XmlIDREF
    Foo foo;

}

input.xml

<?xml version="1.0" encoding="UTF-8"?>
<foo>
    <id>1</id>
    <fooProp>A</fooProp>
    <bar>
        <foo>1</foo>
        <barProp>B</barProp>
    </bar>
    <bar>
        <foo>1</foo>
        <barProp>B</barProp>
    </bar>
</foo>

詳細については

于 2012-12-19T20:31:52.927 に答える