Какова цель minOccurs, nillable и ограничение?

Документация для required говорит:

Если required() true, то свойство Javabean отображается на элемент XML-схемы с помощью minOccurs="1". maxOccurs является "1" для однозначного свойство и "unbounded" для многозначного свойства.

Если required() - false, то свойство Javabean отображается в XML Объявление элемента схемы с помощью minOccurs="0". maxOccurs является "1" для однозначное свойство и "unbounded" для многозначного свойства.

Документация для nillable говорит:

Если nillable() - true, тогда свойство JavaBean отображается в XML Объявление схемы nillable.

<ч/" > Код для xs:complexType:

public class WSData {
    //...

    @XmlElement(required = true, nillable = false)
    public void setMonth(XmlMonthType month) {
        this.month = month;
    }

    public void setUserLogin(String userLogin) {
        this.userLogin = userLogin;
    }
}

Код для xs:simpleType:

@XmlType
@XmlEnum(Integer.class)
public enum XmlMonthType {
    @XmlEnumValue("1")
    JANUARY,
    @XmlEnumValue("2")
    FEBRUARY,
    @XmlEnumValue("3")
    MARCH,
    /* ... months 4 ~9 ... */
    @XmlEnumValue("10")
    OCTOBER,
    @XmlEnumValue("11")
    NOVEMBER,
    @XmlEnumValue("12")
    DECEMBER;
}

Сгенерированная схема XML:

<xs:complexType name="wsData">
  <xs:sequence>
    <xs:element name="month" type="xs:string"/>
    <xs:element minOccurs="0" name="userLogin" type="xs:string"/>
  </xs:sequence>
</xs:complexType>

<xs:simpleType name="xmlMonthType">
  <xs:restriction base="xs:int">
    <xs:enumeration value="1"/>
    <xs:enumeration value="2"/>
    <xs:enumeration value="3"/>
    <!-- ... months 4 ~9 ... -->
    <xs:enumeration value="10"/>
    <xs:enumeration value="11"/>
    <xs:enumeration value="12"/>
  </xs:restriction>
</xs:simpleType>

<ч/" > Факты:

  • Значение по умолчанию для minOccurs равно 1. Таким образом, требуется месяц (должен существовать);
  • В месяце есть ограничение . Таким образом, месяц может иметь только значение, определенное одним из 12 определенных перечислений;
  • Значение по умолчанию для nillable - false. Таким образом, месяц не может иметь пустые значения;
  • XML-схема генерируется правильно.

Проблемы:

  • Принимает абсент месяца (не должен существовать);
  • Принимает любые значения за месяц, например 13 (кроме случаев, когда он не обрабатывается Integer);
  • Принимает пустые значения;

Я не ожидал этих проблем, я что-то упустил?
Если это поведение является правильным, в чем цель required, nillable и xs: restriction

Ответы

Ответ 1

Nillable позволяет пустые значения. Например, если у вас есть Integer или Date, если он не имеет значения, тег XML может быть пустым. Если он не является nillable, но не требуется, элемент XML должен либо существовать с допустимым содержимым, либо вообще не существовать; пустой тег недействителен.

Ответ 2

Сделайте minOccurs 1, чтобы потребовался месяц;

Значение по умолчанию для minOccurs равно единице, поэтому требуется элемент month. Обратите внимание, что minOccurs="0" нужно было добавить в userLogin, чтобы сделать его необязательным.

<xs:complexType name="wsData">
  <xs:sequence>
    <xs:element name="month" type="xs:string"/>
    <xs:element minOccurs="0" name="userLogin" type="xs:string"/>
  </xs:sequence>
</xs:complexType>

Проверить месяц с его сгенерированным ограничением (без XmlAdapter).

Вы можете установить экземпляр Schema в Unmarshaller для подтверждения ввода:

Demo

Следующий код может использоваться для генерации XML-схемы:

package forum9111936;

import java.io.IOException;

import javax.xml.bind.JAXBContext;
import javax.xml.bind.SchemaOutputResolver;
import javax.xml.transform.Result;
import javax.xml.transform.stream.StreamResult;

public class Demo {

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

            @Override
            public Result createOutput(String namespaceUri,
                    String suggestedFileName) throws IOException {
                StreamResult result = new StreamResult(System.out);
                result.setSystemId(suggestedFileName);
                return result;
            }

        };
        jc.generateSchema(sor);
        System.out.println();
    }

}

UPDATE

JAXB RI обычно выбрасывает ValidationEvent серьезности 1 для проблем с конверсией. По умолчанию ValidationEventHandler игнорирует все проблемы с уровнем серьезности менее 2. Обычно это приводит к тому, что значение устанавливается равным нулю. Вы можете переопределить ValidationEventHandler следующим образом:

    unmarshaller.setEventHandler(new ValidationEventHandler() {
        @Override
        public boolean handleEvent(ValidationEvent event) {
            System.out.println(event);
            return event.getSeverity() < ValidationEvent.ERROR;
        }
    });

Однако JAXB RI не отображает события, связанные с преобразованием значений enum (возможная ошибка). Если вы используете EclipseLink JAXB (MOXy) в качестве поставщика JAXB, вы получите исключение, например:

Exception in thread "main" Local Exception Stack: 
Exception [EclipseLink-116] (Eclipse Persistence Services - 2.4.0.qualifier): org.eclipse.persistence.exceptions.DescriptorException
Exception Description: No conversion value provided for the value [13] in field [month/text()].
Mapping: org.eclipse.persistence.oxm.mappings.XMLDirectMapping[month-->month/text()]
Descriptor: XMLDescriptor(forum9111936.WSData --> [])
    at org.eclipse.persistence.exceptions.DescriptorException.noFieldValueConversionToAttributeValueProvided(DescriptorException.java:1052)
    at org.eclipse.persistence.mappings.converters.ObjectTypeConverter.convertDataValueToObjectValue(ObjectTypeConverter.java:140)
    at org.eclipse.persistence.oxm.mappings.XMLDirectMapping.getAttributeValue(XMLDirectMapping.java:287)
    at org.eclipse.persistence.internal.oxm.XMLDirectMappingNodeValue.endElement(XMLDirectMappingNodeValue.java:190)
    at org.eclipse.persistence.oxm.record.UnmarshalRecord.endElement(UnmarshalRecord.java:910)
    at org.eclipse.persistence.internal.oxm.record.XMLStreamReaderReader.parseEvent(XMLStreamReaderReader.java:133)
    at org.eclipse.persistence.internal.oxm.record.XMLStreamReaderReader.parse(XMLStreamReaderReader.java:83)
    at org.eclipse.persistence.internal.oxm.record.XMLStreamReaderReader.parse(XMLStreamReaderReader.java:72)
    at org.eclipse.persistence.internal.oxm.record.SAXUnmarshaller.unmarshal(SAXUnmarshaller.java:838)
    at org.eclipse.persistence.oxm.XMLUnmarshaller.unmarshal(XMLUnmarshaller.java:626)
    at org.eclipse.persistence.jaxb.JAXBUnmarshaller.unmarshal(JAXBUnmarshaller.java:472)
    at forum9111936.Demo2.main(Demo2.java:30)

Дополнительная информация

Ответ 3

Цели required и minOccurs не вводят в заблуждение, проблема в том, что проверка схемы не включена. Просто включите SchemaValidation в WebService и порядок XmlType:

Веб-сервис:

@javax.jws.WebService
@org.jboss.ws.annotation.SchemaValidation(enabled = true)
public class WebServiceClass {

    @javax.jws.WebMethod
    public WSResponseData webServiceMethod() {
        //...
    }
}

XmlType:

@javax.xml.bind.annotation.XmlType(propOrder = {"field1", "field2", "field3"})
public class WSData {
    //...

    private String field1;

    private Long field2;

    private XmlMonthType field3;

    //...
}

Ответ 4

Действительно, кажется, что SOAP-конверты не проверяются WSDL, ни на сервере, ни на клиенте.

И я думаю, что так лучше. Валидация потребляет ресурсы и время. Для большинства WebServices все, что нам нужно с WSDL, - предоставить клиенту определения, чтобы иметь возможность разговаривать с WebService, а не устанавливать ограничения, например, если свойство может быть или не быть нулевым.

Когда сервер и клиент находятся на одном ПК, Axis2 добавляет 3-5 milisecs накладных расходов по сравнению с внутренними вызовами. Нежелательная проверка только увеличит накладные расходы. Мне лучше делать операции, генерируя исключение и вручную проверяя, что действительно нужно.

Ответ 5

Я просмотрел структуру XSD, которую вы опубликовали, и обнаружил, что вы не использовали сложный тип, который вы определили за месяц, Insteqad обрабатывает его по java-коду, вы можете упомянуть, что все в XSD, а также minOccur в месяц - элемент вашего сложного типа отсутствует. Пожалуйста, используйте любой встроенный генератор XSD для java-генератора, он выполнит всю задачу создания Java файла для вас.

The convention are as follow - 

MinOccur = 0 --> the element can be abscent in input , and can be present

MinOccur = 1 --> the element must be there in input
(but if you use it, then your java generated member will be of list type - list of ENUM for Int )

If you don't write MinOccur in attributes - then it makes the element mandatory , ( then you java generated member will be of simply ENUM for int )


MaxOccur = 1 --> minimum one element can be there in input 
(but if you use it, then your java generated member will be of list type - list of ENUM for Int )


MaxOccur = unbound --> only one element can be there in input 
(if you use it, then your java generated member will be of list type - list of ENUM for Int )

   <xs:complexType name="wsData">
  <xs:sequence>
    <xs:element name="month" type="xmlMonthType" minOccurs="1" nillable="false" />
    <xs:element name="userLogin" type="xs:string" minOccurs="0" />
  </xs:sequence>
</xs:complexType>

<xs:simpleType name="xmlMonthType">
  <xs:restriction base="xs:int">
    <xs:enumeration value="1"/>
    <xs:enumeration value="2"/>
    <xs:enumeration value="3"/>
    <!-- ... months 4 ~9 ... -->
    <xs:enumeration value="10"/>
    <xs:enumeration value="11"/>
    <xs:enumeration value="12"/>
  </xs:restriction>
</xs:simpleType>