6

我有一个代表 XML 模式的 POJO 对象树。这是使用以下jaxbant 脚本创建的。

我想针对缺少属性的架构验证根 POJO 及其子实体。

我的代码如下:(省略了 try/catch 块,灵感来自 SO question How to validate against schema in JAXB 2.0 without marshalling?

public boolean validateAgainstSchema(Pojo pojo)
{
        JAXBContext jc;
        jc = JAXBContext.newInstance(Pojo.class);
        SchemaFactory sf = SchemaFactory.newInstance(XMLConstants.W3C_XML_SCHEMA_NS_URI);
        Schema schema = sf.newSchema(new ClassPathResource("schema.xsd").getFile());

        Marshaller marshaller = jc.createMarshaller();
        marshaller.setSchema(schema);
        marshaller.marshal(schema, new DefaultHandler());
        return true;
}

我的一个属性 ( pojo.childEntity.someAttribute) 是date

XSD

<xsd:attribute name="some_date" use="required">
  <xsd:simpleType>
    <xsd:restriction base="xsd:date" />
  </xsd:simpleType>
</xsd:attribute>

爪哇

@XmlAttribute(name = "someDate", required = true)
protected XMLGregorianCalendar someDate;

它从一个java.util.Date它从另一个 POJO(使用 Hibernate 映射

private static final XMLGregorianCalendar dateToCalendar(Date date)
{
    if (date == null)
        return null;
    try
    {
        GregorianCalendar c = new GregorianCalendar();
        c.setTime(date);

        return DatatypeFactory.newInstance()
                .newXMLGregorianCalendar(c);
    }
    catch (DatatypeConfigurationException e)
    {
        e.printStackTrace();
        return null;
    }

}

例外是:

javax.xml.bind.MarshalException
 - with linked exception:
[org.xml.sax.SAXParseException: cvc-datatype-valid.1.2.1: '2001-05-11T00:00:00.000+02:00' is not a valid value for 'date'.]

这看起来像 JAXB 试图为必须只携带日期的字段设置日期和时间,而 XMLGregorianCalendar 只是一个日期时间容器。

问题是:是什么导致了错误?怎么修?

4

2 回答 2

3

默认情况下,XMLGregorianCalendar属性的输出将基于您创建它的方式。如果填充时间部分,则时间部分将输出到 XML。您可以调用该getXMLSchemaType()方法来查看对应的 XML 表示形式是什么:

您可以使用@XmlSchemaType注释来覆盖表示。

Java 模型(根)

下面是一个具有 3 个XMLGregorianCalendar字段的对象。3日我将使用@XmlSchemaType注解来指定模式类型。

import javax.xml.bind.annotation.*;
import javax.xml.datatype.XMLGregorianCalendar;

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

    protected XMLGregorianCalendar default1;

    protected XMLGregorianCalendar default2;

    @XmlSchemaType(name="date")
    protected XMLGregorianCalendar schemaTypeDate;

}

演示代码

在下面的演示代码中,我们将创建 2 个XMLGregorianCalendar. 一种将具有模式类型date,另一种dateTime。默认情况下,这是编组为 XML 时使用的 XML 表示。在该schemaTypeDate字段上,我们将使用@XmlSchemaType注释来覆盖默认值。

import javax.xml.bind.*;
import javax.xml.datatype.*;

public class Demo {

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

        DatatypeFactory df = DatatypeFactory.newInstance();
        XMLGregorianCalendar date  = df.newXMLGregorianCalendar("2013-07-03");
        XMLGregorianCalendar dateTime = df.newXMLGregorianCalendar("1999-12-31T23:59:00");

        Root root = new Root();
        root.default1 = date;
        root.default2 = dateTime;
        root.schemaTypeDate = dateTime;

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

}

输出

<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<root>
    <default1>2013-07-03</default1>
    <default2>1999-12-31T23:59:00</default2>
    <schemaTypeDate>1999-12-31</schemaTypeDate>
</root>

更新

好的,因为我有 looooooooooooooooooooooooooooooooots 的 XmlGregorianCalendars 有没有办法告诉 XJC 将 xmlSchemaType 属性添加到所有 XGC?

xsd:date当模式类型是 XML 模式类型中的一种,即or时, JAXB 将为您执行此操作,xsd:dateTime但当您扩展了其中一种类型时则不会。

XML 模式 (schema.xsd)

<?xml version="1.0" encoding="UTF-8"?>
<schema xmlns="http://www.w3.org/2001/XMLSchema">
    <complexType name="root">
        <sequence>
            <element name="dateElement" type="date" />
            <element name="dateTimeElement" type="dateTime" />
            <element name="extendedDateElement">
                <simpleType>
                    <restriction base="date" />
                </simpleType>
            </element>
        </sequence>
        <attribute name="dateAttribute" type="date" />
        <attribute name="dateTimeAttribute" type="dateTime" />
        <attribute name="extendedDateAttribute">
            <simpleType>
                <restriction base="date" />
            </simpleType>
        </attribute>
    </complexType>
</schema>

@XmlAccessorType(XmlAccessType.FIELD)
@XmlType(name = "root", propOrder = {
    "dateElement",
    "dateTimeElement",
    "extendedDateElement"
})
public class Root {

    @XmlElement(required = true)
    @XmlSchemaType(name = "date")
    protected XMLGregorianCalendar dateElement;

    @XmlElement(required = true)
    @XmlSchemaType(name = "dateTime")
    protected XMLGregorianCalendar dateTimeElement;

    @XmlElement(required = true)
    protected XMLGregorianCalendar extendedDateElement;

    @XmlAttribute(name = "dateAttribute")
    @XmlSchemaType(name = "date")
    protected XMLGregorianCalendar dateAttribute;

    @XmlAttribute(name = "dateTimeAttribute")
    @XmlSchemaType(name = "dateTime")
    protected XMLGregorianCalendar dateTimeAttribute;

    @XmlAttribute(name = "extendedDateAttribute")
    protected XMLGregorianCalendar extendedDateAttribute;

}
于 2013-07-03T10:01:35.793 回答
1

如果日期是 '2001-05-11T00:00:00.000+02:00' 使用

<xsd:restriction base="xsd:dateTime" />
于 2013-07-03T10:00:19.133 回答