3

假设我有类人:

class Person{
  String firstName;
  String lastName;
  String email;
}

XML 具有以下格式:

<person>
 <firstName value="asd" /> 
 <lastName value="bcd" />
 <email value="qwe" />
</person>

我可以为 FirstNameAdapter、LastNameAdapter、EmailAdapter 中的每个字段使用自己的 XmlAdapter 实现来解组/编组此类。如您所见,每个字段都以类似的方式表示 - 字段名称作为 xml 元素,字段值作为元素的属性。是否可以创建“通用”适配器,我可以将字段名称传输到该适配器并提取该字段的值?

PS 我知道 MOXy JAXB 实现,但我想知道是否可以通过参考 JAXB 实现来实现。

谢谢!

4

2 回答 2

5

你可以使用XmlAdapter这样的:

import java.io.*;
import javax.xml.bind.*;
import javax.xml.bind.annotation.*;
import javax.xml.bind.annotation.adapters.*;

@XmlType
class Valued {
    @XmlAttribute(name="value")
    public String value;
}

class ValuedAdapter extends XmlAdapter<Valued, String> {
    public Valued marshal(String s) {
        Valued v = new Valued();
        v.value = s;
        return v;
    }
    public String unmarshal(Valued v) {
        return v.value;
    }
}

@XmlRootElement
class Person {

    @XmlJavaTypeAdapter(ValuedAdapter.class)
    @XmlElement
    String firstName;

    @XmlJavaTypeAdapter(ValuedAdapter.class)
    @XmlElement
    String lastName;

}

class SO12928971 {
    public static void main(String[] args) throws Exception {
        Person p = new Person();
        p.firstName = "John";
        p.lastName = "Doe";
        JAXBContext jc = JAXBContext.newInstance(Person.class);
        StringWriter sw = new StringWriter();
        jc.createMarshaller().marshal(p, sw);
        String xml = sw.toString();
        System.out.println(xml);
        StringReader sr = new StringReader(xml);
        p = (Person)jc.createUnmarshaller().unmarshal(sr);
        assert "John".equals(p.firstName);
        assert "Doe".equals(p.lastName);
    }
}

这里的想法是 XML Schema 以及 JAXB 在元素名称内容类型之间有明显的区别,尽管许多文档在这两者之间具有明确的一一对应关系。所以在上面的代码中,类型 Valued描述了具有value属性的东西,而不考虑元素名称。您要序列化的成员被注释为该注释中@XmlElement不包含名称。因此,它们将生成名称源自成员名称的元素。注释将@XmlJavaTypeAdapter导致序列化程序将这些成员视为它们的类型 vere Valued。这就是他们的 XML 内容类型。

上述代码的架构如下所示:

<xs:schema version="1.0" xmlns:xs="http://www.w3.org/2001/XMLSchema">

  <xs:element name="person" type="person"/>

  <xs:complexType name="person">
    <xs:sequence>
      <xs:element name="firstName" type="valued" minOccurs="0"/>
      <xs:element name="lastName" type="valued" minOccurs="0"/>
    </xs:sequence>
  </xs:complexType>

  <xs:complexType name="valued">
    <xs:sequence/>
    <xs:attribute name="value" type="xs:string"/>
  </xs:complexType>
</xs:schema>
于 2012-10-17T08:49:47.610 回答
0

注意: 我是EclipseLink JAXB (MOXy)负责人,也是JAXB (JSR-222)专家组的成员。

PS 我知道 MOXy JAXB 实现,但我想知道是否可以通过参考 JAXB 实现来实现。

为了比较起见,我添加了如何使用 EclipseLink JAXB (MOXy) 的@XmlPath扩展来完成此操作。

package forum12928971;

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

@XmlRootElement
@XmlAccessorType(XmlAccessType.FIELD)
class Person{

    @XmlPath("firstName/@value")
    String firstName;

    @XmlPath("lastName/@value")
    String lastName;

    @XmlPath("email/@value")
    String email;

}

jaxb.properties

要将 MOXy 指定为您的 JAXB 提供程序,您需要添加一个jaxb.properties在与您的域模型相同的包中调用的文件,其中包含以下条目:

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

演示

package forum12928971;

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

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

        Unmarshaller unmarshaller = jc.createUnmarshaller();
        File xml = new File("src/forum12928971/input.xml");
        Person person = (Person) unmarshaller.unmarshal(xml);

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

}

输入.xml/输出

<?xml version="1.0" encoding="UTF-8"?>
<person>
   <firstName value="asd"/>
   <lastName value="bcd"/>
   <email value="qwe"/>
</person>
于 2012-10-17T10:18:48.867 回答