SAXParseException:value不是'date'的有效值

usr*_*ΛΩΝ 6 java datetime xsd jaxb xjc

我有一个POJO对象树,代表一个XML Schema.这是使用以下jaxbant脚本创建的.

我想根据缺少属性的模式验证根POJO及其子实体.

我的代码如下:(省略了try/catch块,受SO问题的启发如何在没有编组的情况下验证JAXB 2.0中的模式?)

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;
}
Run Code Online (Sandbox Code Playgroud)

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

XSD

<xsd:attribute name="some_date" use="required">
  <xsd:simpleType>
    <xsd:restriction base="xsd:date" />
  </xsd:simpleType>
</xsd:attribute>
Run Code Online (Sandbox Code Playgroud)

Java的

@XmlAttribute(name = "someDate", required = true)
protected XMLGregorianCalendar someDate;
Run Code Online (Sandbox Code Playgroud)

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;
    }

}
Run Code Online (Sandbox Code Playgroud)

例外是:

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'.]
Run Code Online (Sandbox Code Playgroud)

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

问题是:导致错误的原因是什么?怎么修?

bdo*_*han 3

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

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

Java 模型(根)

下面是一个具有 3 个字段的对象XMLGregorianCalendar。第三,我将使用@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;

}
Run Code Online (Sandbox Code Playgroud)

演示代码

在下面的演示代码中,我们将创建 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);
    }

}
Run Code Online (Sandbox Code Playgroud)

输出

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;

}
Run Code Online (Sandbox Code Playgroud)

更新

好的,既然我有 loooooooooooooooooooooooooooooooooots 的 XmlGregorianCalendars,有没有办法告诉 XJC 将 xmlSchemaType 属性添加到所有 XGC?

xsd:date当模式类型是 XML 模式类型(即或 )中构建的类型之一时,JAXB 将为您执行此操作xsd:dateTime,但当您扩展了这些类型之一时则不会。

XML 架构 (schema.xsd)

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);
    }

}
Run Code Online (Sandbox Code Playgroud)

@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;

}
Run Code Online (Sandbox Code Playgroud)