2

ドメイン オブジェクトから POJO を生成する Web アプリケーションがあります。ドメイン オブジェクトの 1 つにマップが含まれており、JAXB は次のスキーマを生成します。

<xs:element name="persons">
  <xs:complexType>
      <xs:sequence>
          <xs:element name="entry" minOccurs="0" maxOccurs="unbounded">
              <xs:complexType>
                 <xs:sequence>
                    <xs:element name="key" minOccurs="0" type="xs:string"/>
                    <xs:element name="value" minOccurs="0" type="person"/>
                 </xs:sequence>
              </xs:complexType>
          </xs:element>
      </xs:sequence>
   </xs:complexType>
</xs:element>

これはHashMap<String, Person>人から生成されます:

@XmlAccessorType(XmlAccessType.FIELD)
@XmlType(name = "personConfiguration", propOrder = {
    "persons",
})
@XmlRootElement(name = "personConfiguration")
public class PersonConfiguration
{

    @XmlElement(required = true)
    protected PersonConfiguration.Persons persons;

    /**
     * Gets the value of the persons property.
     * 
     * @return
     *     possible object is
     *     {@link PersonConfiguration.Persons }
     *     
     */
    public PersonConfiguration.Persons getPersons() {
        return persons;
    }

    /**
     * Sets the value of the persons property.
     * 
     * @param value
     *     allowed object is
     *     {@link PersonConfiguration.Persons }
     *     
     */
    public void setPersons(PersonConfiguration.Persons value) {
        this.persons = value;
    }

    @XmlAccessorType(XmlAccessType.FIELD)
    @XmlType(name = "", propOrder = {
        "entry"
    })
    public static class Persons
    {

        protected List<PersonConfiguration.Persons.Entry> entry;

        /**
         * Gets the value of the entry property.
         * 
         * <p>
         * This accessor method returns a reference to the live list,
         * not a snapshot. Therefore any modification you make to the
         * returned list will be present inside the JAXB object.
         * This is why there is not a <CODE>set</CODE> method for the entry property.
         * 
         * <p>
         * For example, to add a new item, do as follows:
         * <pre>
         *    getEntry().add(newItem);
         * </pre>
         * 
         * 
         * <p>
         * Objects of the following type(s) are allowed in the list
         * {@link PersonConfiguration.Persons.Entry }
         * 
         * 
         */
        public List<PersonConfiguration.Persons.Entry> getEntry() {
            if (entry == null) {
                entry = new ArrayList<PersonConfiguration.Persons.Entry>();
            }
            return this.entry;
        }


        @XmlAccessorType(XmlAccessType.FIELD)
        @XmlType(name = "", propOrder = {
            "key",
            "value"
        })
        public static class Entry
        {

            protected String key;
            protected Person value;

            /**
             * Gets the value of the key property.
             * 
             * @return
             *     possible object is
             *     {@link String }
             *     
             */
            public String getKey() {
                return key;
            }

            /**
             * Sets the value of the key property.
             * 
             * @param value
             *     allowed object is
             *     {@link String }
             *     
             */
            public void setKey(String value) {
                this.key = value;
            }

            /**
             * Gets the value of the value property.
             * 
             * @return
             *     possible object is
             *     {@link Person }
             *     
             */
            public Person getValue() {
                return value;
            }

            /**
             * Sets the value of the value property.
             * 
             * @param value
             *     allowed object is
             *     {@link Person }
             *     
             */
            public void setValue(Person value) {
                this.value = value;
            }

        }

    }

}

ご覧のとおり、JAXB はこの追加レベルの間接エントリ -> キー、値を追加しました。パズルの他のピースは、Spring MVC、JSON オブジェクトを使用した REST 呼び出しです。

現在、XML ベースの REST 呼び出しは上記のオブジェクト スキーマで正常に動作しますが、同じスキーマで JSON メッセージを使用して同じ呼び出しを送信すると、JSONMappingException が発生します。

なぜこれが起こっているのでしょうか?

4

1 に答える 1

2

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

異なる XML および JSON バインディング プロバイダーを使用する場合、XML および JSON 表現の一貫性を維持することは困難です。以下は、XML および JSON プロバイダーの両方として MOXy を使用し、すべてのマッピング情報を JAXB アノテーションとして提供して、これを単純化する方法の例です。

persons以下は、フィールドが質問から XML スキーマ フラグメントを生成するドメイン オブジェクトのサンプルです。

package forum13784163;

import java.util.Map;

import javax.xml.bind.annotation.*;

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

    Map<String, Person> persons;

}

Person以下は、クラスがどのように見えるかのサンプルです。idフィールドを XML 属性にマップした方法に注意してください。

package forum13784163;

import javax.xml.bind.annotation.*;

@XmlAccessorType(XmlAccessType.FIELD)
public class Person {

    @XmlAttribute
    int id;

    String name;

    int age;

}

jaxb.properties

MOXy を JAXB プロバイダーとして使用するにはjaxb.properties、次のエントリを使用してドメイン モデルと同じパッケージに呼び出されるファイルを含める必要があります ( http://blog.bdoughan.com/2011/05/specifying-eclipselink-moxy-asを参照)。 -your.html )。

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

入力.json

以下は、JSON バインディング プロバイダーとして MOXy を使用した場合の JSON 表現です。

{
   "persons" : {
      "entry" : [ {
         "key" : "Jane",
         "value" : {
            "id" : 123,
            "name" : "Jane",
            "age" : 30
         }
      } ]
   }
}

デモ

以下のデモ コードでは、JSON がオブジェクトにアンマーシャリングされ、同じオブジェクトが XML にマーシャリングされます。これは、JAXBContext1 セットのメタデータを含むものから実行されます。

package forum13784163;

import javax.xml.bind.*;
import javax.xml.transform.stream.StreamSource;
import org.eclipse.persistence.jaxb.UnmarshallerProperties;

public class Demo {

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

        Unmarshaller unmarshaller = jc.createUnmarshaller();
        unmarshaller.setProperty(UnmarshallerProperties.MEDIA_TYPE, "application/json");
        unmarshaller.setProperty(UnmarshallerProperties.JSON_INCLUDE_ROOT, false);
        StreamSource json = new StreamSource("src/forum13784163/input.json");
        Root root = unmarshaller.unmarshal(json, Root.class).getValue();

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

}

出力

以下は、結果の XML です。

<?xml version="1.0" encoding="UTF-8"?>
<root>
   <persons>
      <entry>
         <key>Jane</key>
         <value id="123">
            <name>Jane</name>
            <age>30</age>
         </value>
      </entry>
   </persons>
</root>

詳細については

于 2012-12-09T12:29:23.943 に答える