XML Schema nillable=”true” vs minOccurs=”0″



XML Schema nillable=”true” vs minOccurs=”0″

出处:http://www.dimuthu.org/blog/2008/08/18/xml-schema-nillabletrue-vs-minoccurs0/

 

【前言】一个月前研究过nillable=”true”和minOccurs=”0″的区分,没有想到啊,今天就记不得了,看来真是好记性不如烂笔头啊,况且我还么的好记性。。。。

 

【总结】英文一眼看不出结论,说点汉语直接些,别说我土。。。

 

nillable=”true”:该元素的值可以为空,但是该元素不能省略,也就是说,只能:

<minzero xsi:nil=”true”><minzero>
而不能直接将minzero这个元素去掉

(注意:xsi:nil=”true”或者xsi:nil=”1″应该就是表示这个元素为空)

 

minOccurs=”0″:该元素可以直接被省略掉,但是不能让该元素的值为空,也就是说,不能:

<minzero xsi:nil=”true”><minzero>


 

 

In a WSDL, XML Schema is the section where it define the message format for each operations, which eventually become the real API that users are interested. And it is the most tricky part of the WSDL. Nowadays there are many tools that you can design and use WSDLs without any needs in knowing the meaning of a single line of the WSDL. But there are situations that you may find it is better you have some knowledge in XML Schema section and in WSDL overall.
For this post I m taking a simple example of use of nillable=”true” and minOccurs=”0″. Take the following example.

<xs:element name=”myelements”>
<xs:complexType>
<xs:sequence>
<xs:element name=”nonboth” type=”xs:string”/>
<xs:element minOccurs=”0″ name=”minzero” type=”xs:int”/>
<xs:element name=”nilint” nillable=”true” type=”xs:int”/>
<xs:element name=”nilstring” nillable=”true” type=”xs:string”/>
<xs:element minOccurs=”0″ name=”minzeronil” nillable=”true” type=”xs:string”/>
</xs:sequence>
</xs:complexType>
</xs:element>
Just ignore the meaning of what nillable and minOccurs attributes for now. You can safely say the following XML is valid for the above Schema.

<myelements>
<nonboth>i can’t be either nil nor skipped<nonboth>
<minzero>3<minzero>
<nilint><nilint>
<nilstring>i can have null, but i cant skipeed</nilstring>
<minzeronil>i can be skipped and have the nil value<minzeronil>
</myelements>
Take the first element ‘nonboth’ in the schema, It has not any minOccurs or nillable attribute. By default minOccurs equal to 1 and nillable equal to false. That mean it can’t have nil value nor it can not be removed from the xml.

Is that making an element nil and removing the element from the XML is same? No. Take the second element in the schema ‘minzerostring’. There you have minOccurs =”0″ but there are no nillable=”true”, mean it is non-nillable. The idea is whenever you don’t want that element in your xml, you can’t have the element keeping empty like

<minzero xsi:nil=”true”><minzero>
But you can remove the whole element from the XML (since it is minOccurs=0).
The opposite of the above scenario is nillable=”true” but minOccurrs != 0. Check the ‘nilint’ element in the schema. There you can’t skip the element ‘nilint’, you have to have the element <nilint/> but it can hold a nil value.

<nilint xsi:nil=”true”></nilint>

or simply

<nilint xsi:nil=”true”/>
Note that the correct way to declare the nil element is,
<nilint xsi:nil=”true” xmlns:xsi=”http://www.w3.org/2001/XMLSchema-instance”/>
You can understand why when we look at the third element ‘nilstring’. Say you set message the following element
<nilstring></nilstring>
You can say that this is not nil, this is an empty string. In fact an empty string is nil in some other language, But if we take XML Schema as a language, then for someone to be nil, it have to have the xsi:nil attribute set to “true” or “1″.
So going back to the ‘minzero’ which is non-nillable, by theory you should be able to write the following xml,

<minzero/>
Since you don’t have that xsi:nil=”1″ this is not a nil value, so the condition nillable=”false” condition is preserved. But unlike for string when you set an empty element for an integer, it doesn’t sound correct. So in practice whenever some schema says non-nillable you should set some valid value.
The last one is ‘minzeronil’ element which is both nillable=”true” and minOccurs=”0″. Whenever you don’t need to set a value for this element, you have the choice of either skipping the element or setting the value of the element to nil. It is obvious rather than setting a nil value it is better you just skip the element to make the XML shorter. This is really needed specially in web services where you need the payload to be minimum as much as possible.

Say you have to prepare the XML and you don’t have valid values for any of the element. So this can be the optimum XML you can create.

<myelements xmlns:xsi=”http://www.w3.org/2001/XMLSchema-instance”/>
<nonboth><nonboth>
<nilint xsi:nil=”1″/>
<nilstring xsi:nil=”1″>
</myelements>
Read this nice article in developer works on nillable=”true” and minOccurs=”0″ for more.