- 浏览: 263864 次
- 性别:
- 来自: 天津
文章分类
- 全部博客 (180)
- J2SE (15)
- J2EE (7)
- Struts (2)
- Hibernate (18)
- Spring (22)
- ssh (2)
- HTML (1)
- JS (10)
- CSS (3)
- jQuery (15)
- extJS (0)
- FLEX (14)
- Web service (7)
- Axis2 (3)
- lucene2.0 (9)
- maven (3)
- JBoss (0)
- WebLogic (0)
- Linux (8)
- Android (0)
- SiteMesh (1)
- JPA (0)
- JBPM (1)
- xFire (3)
- XML (4)
- apache (4)
- bingo专区 (0)
- 加密解密算法研究 (1)
- 数据库 (16)
- 资料 (1)
- 经验 (4)
- 健康 (1)
- 技术交流区 (1)
- 交友区 (0)
- jquery 插件区 (1)
最新评论
-
forrest_lv:
很有帮助
Axis2 XFire CXF 比较 -
kuloujianzun:
...
flex DataGrid -
zx.peng:
大哥,请教一下...
下面在别的地方引入:(在什么地方、怎么 ...
FLEX下拉多选框 -
elvislee030:
试试去,正好要找这个组件。。先谢过
FLEX下拉多选框 -
futrueboy:
如果你发到网上来,请翻译的好点
Apache Mina的学习应用(三)
一、为什么使用WSDL?
像Internet协议之类的标准有没有为权威所利用,或者人们这样看待它是因为顺之所获的好处远远超出了代价?曾经有许多试图建立的标准都流产了。有时候,那些还没有普遍使用的标准甚至由法令或政府规定强行推出:Ada语言就是一例。
我相信正是跟随标准所带来的好处使它广泛接受。例如,对于铁路服务来说,真正重要的是,不同公司所铺设的铁路结合到一起,或者是来自好几个公司的产品协调的工作在一起。几家大的企业合力建立了SOAP标准。Web Service描述语言(WSDL)向这种Web Service的提供商和用户推出了方便的协调工作的方法,使我们能更容易的获得SOAP的种种好处。几家公司的铁道并在一起不算什么难事,他们所需遵循的只是两轨间的标准距离。对Web Service来说,这要复杂得多。我们必须先制定出指定接口的标准格式。
曾经有人说SOAP并不真需要什么接口描述语言。如果SOAP是交流纯内容的标准,那就需要一种语言来描述内容。SOAP消息确实带有某些类型信息,因此SOAP允许动态的决定类型。但不知道一个函数的函数名、参数的个数和各自类型,怎么可能去调用这个函数呢?没有WSDL,我可以从必备文档中确定调用语法,或者检查消息。随便何种方法,都必须有人参与,这个过程可能会有错。而使用了WSDL,我就可以通过这种跨平台和跨语言的方法使Web Service代理的产生自动化。就像COM和CORBA的IDL文件,WSDL文件由客户和服务器约定。
注意由于WSDL设计成可以绑定除SOAP以外的其他协议,这里我们主要关注WSDL在HTTP上和SOAP的关系。同样,由于SOAP目前主要用来调用远程的过程和函数,WSDL支持SOAP传输的文档规范。WSDL 1.1已经作为记录递交给W3C(见http://www.w3.org/TR/wsdl.html)
二、WSDL文档结构
若要理解XML文档,将之看作块状图表非常有用。下图以XML的文档形式说明了WSDL的结构,它揭示了WSDL文档五个栏之间的关系。
WSDL文档可以分为两部分。顶部分由抽象定义组成,而底部分则由具体描述组成。抽象部分以独立于平台和语言的方式定义SOAP消息,它们并不包含任何随机器或语言而变的元素。这就定义了一系列服务,截然不同的网站都可以实现。随网站而异的东西如序列化便归入底部分,因为它包含具体的定义。
l 抽象定义
Types
独立与机器和语言的类型定义
Messages
包括函数参数(输入与输出分开)或文档描述
PortTypes
引用消息部分中消息定义来描述函数签名(操作名、输入参数、输出参数)
2 具体描述
Bindings
PortTypes部分的每一操作在此绑定实现
Services
确定每一绑定的端口地址
下面的图中,箭头连接符代表文档不同栏之间的关系。点和箭头代表了引用或使用关系。双箭头代表"修改"关系。3-D的箭头代表了包含关系。这样,各Messages栏使用Types栏的定义,PortTypes栏使用Messages栏的定义;Bindings栏引用了PortTypes栏,Services栏引用Bindings栏,PortTypes和Bindings栏包含了operation元素,而Services栏包含了port元素。PortTypes栏里的operation元素由Bindings栏里的operation元素进一步修改或描述。
在此背景中,我将使用标准的XML术语来描述WSDL文档。Element是指XML的元素,而"attribute"指元素的属性。于是:
<element attribute="attribute-value">contents</element>
内容也可能由一个或多个元素以递归的方式组成。根元素是所有元素之中最高级的元素。子元素总是从属于另一个元素,父元素。
注意,文档之中可能只有一个Types栏,或根本没有。所有其他的栏可以只有零元素、单元素或是多元素。WSDL的列表要求所有的栏以固定的顺序出现:import, types, message, portType, binding, service。所有的抽象可以是单独存在于别的文件中,也可以从主文档中导入。
图一:抽象定义和具体定义
三、WSDL文件示例
让我们来研究一下WSDL文件,看看它的结构,以及如何工作。请注意这是一个非常简单的WSDL文档实例。我们的意图只是说明它最显著的特征。以下的内容中包括更加详细的讨论。
<?xml version=“1.0” encoding=“UTF-8” ?>
<definitions name=“FooSample”
targetNamespace=“http://tempuri.org/wsdl/”
xmlns:wsdlns=“http://tempuri.org/wsdl/”
xmlns:typens=“http://tempuri.org/xsd”
xmlns:xsd=“http://www.w3.org/2001/XMLSchema”
xmlns:soap=“http://schemas.xmlsoap.org/wsdl/soap/”
xmlns:stk=“http://schemas.microsoft.com/soap-toolkit/wsdl-extension”
xmlns=“http://schemas.xmlsoap.org/wsdl/”>
<types>
<schema targetNamespace=“http://tempuri.org/xsd”
xmlns=“http://www.w3.org/2001/XMLSchema”
xmlns:SOAP-ENC=“http://schemas.xmlsoap.org/soap/encoding/”
xmlns:wsdl=“http://schemas.xmlsoap.org/wsdl/”
elementFormDefault=“qualified” >
</schema>
</types>
<message name=“Simple.foo”>
<part name=“arg” type=“xsd:int”/>
</message>
<message name=“Simple.fooResponse”>
<part name=“result” type=“xsd:int”/>
</message>
<portType name=“SimplePortType”>
<operation name=“foo” parameterOrder=“arg” >
<input message=“wsdlns:Simple.foo”/>
<output message=“wsdlns:Simple.fooResponse”/>
</operation>
</portType>
<binding name=“SimpleBinding” type=“wsdlns:SimplePortType”>
<stk:binding preferredEncoding=“UTF-8” />
<soap:binding style=“rpc”
transport=“http://schemas.xmlsoap.org/soap/http”/>
<operation name=“foo”>
<soap:operation soapAction=“http://tempuri.org/action/Simple.foo”/>
<input>
<soap:body use=“encoded” namespace=“http://tempuri.org/message/”
encodingStyle=“http://schemas.xmlsoap.org/soap/encoding/” />
</input>
<output>
<soap:body use=“encoded” namespace=“http://tempuri.org/message/”
encodingStyle=“http://schemas.xmlsoap.org/soap/encoding/” />
</output>
</operation>
</binding>
<service name=“FOOSAMPLEService”>
<port name=“SimplePort” binding=“wsdlns:SimpleBinding”>
<soap:address location=“http://carlos:8080/FooSample/FooSample.asp”/>
</port>
</service>
</definitions>
以下是该实例文档的总述:稍后我将详细讨论每一部分的细节。
第一行申明该文档是XML。尽管这并不是必需的,但它有助于XML解析器决定是否解析WSDL文件或只是报错。第二行是WSDL文档的根元素:<definitions>。一些属性附属于根元素,就像<schema>子元素对于<types>元素。
<types>元素包含了Types栏。如果没有需要声明的数据类型,这栏可以缺省。在WSDL范例中,没有应用程序特定的types声明,但我仍然使用了Types栏,只是为了声明schema namespaces。
<message>元素包含了Messages栏。如果我们把操作看作函数,<message>元素定义了那个函数的参数。<message>元素中的每个<part>子元素都和某个参数相符。输入参数在<message>元素中定义,与输出参数相隔离--输出参数有自己的<message>元素。兼作输入、输出的参数在输入输出的<message>元素中有它们相应的<part>元素。输出<message>元素以”Response”结尾,就像以前所用的”fooResponse”。每个<part>元素都有名字和类型属性,就像函数的参数有参数名和参数类型。
用于交换文档时,WSDL允许使用<message>元素来描述交换的文档。
<part>元素的类型可以是XSD基类型,也可以是SOAP定义类型(soapenc)、WSDL定义类型(wsdl)或是Types栏定义的类型。
一个PortTypes栏中,可以有零个、单个或多个<portType>元素。由于抽象PortType定义可以放置在分开的文件中,在某个WSDL文件中没有<portType>元素是可能的。上面的例子里只是用了一个<portType>元素。而一个<portType>元素可在<operation>元素中定义一个或是多个操作。示例仅使用了一个名为”foo”的<operation>元素。这和某个函数名相同。<operation>元素可以有一个、两个、三个子元素:<input>, <output> 和<fault>元素。每个<input>和<output>元素中的消息都引用Message栏中的相关的<message>元素。这样,示例中的整个<portType>元素就和以下的C函数等效:
int foo(int arg);
这个例子足见XML和C相比要冗长的多。(包括<message>元素,XML在示例中共使用了12行代码来表达相同的单行函数声明。)
Bindings栏可以有零个、一个或者多个<binding>元素。它的意图是制定每个<operation>通过网络调用和回应。Services栏同样可以有零个、一个、多个<service>元素。它还包含了<port>元素,每个<port>元素引用一个Bindings栏里的<binding>元素。Bindings和Services栏都包含WSDL文档。
Namespace
<definitions>和子节点<schema>都是namespace属性:
<definitions name=“FooSample”
targetNamespace=“http://tempuri.org/wsdl/”
xmlns:wsdlns=“http://tempuri.org/wsdl/”
xmlns:typens=“http://tempuri.org/xsd”
xmlns:xsd=“http://www.w3.org/2001/XMLSchema”
xmlns:soap=“http://schemas.xmlsoap.org/wsdl/soap/”
xmlns:stk=“http://schemas.microsoft.com/soap-toolkit/wsdl-extension”
xmlns=“http://schemas.xmlsoap.org/wsdl/”>
<types>
<schema targetNamespace=“http://tempuri.org/xsd”
xmlns=“http://www.w3.org/2001/XMLSchema”
xmlns:SOAP-ENC=“http://schemas.xmlsoap.org/soap/encoding/”
xmlns:wsdl=“http://schemas.xmlsoap.org/wsdl/”
elementFormDefault=“qualified” >
</schema>
</types>
每个namespace属性都声明了一个缩略语,用在文档中。例如”xmlns:xsd”就为 http://www.w3.org/2001/XMLSchema定义了一个缩略语(xsd)。这就允许对该namespace的引用只需简单的在名字前加上前缀就可以了,如:”xsd:int”中的”xsd”就是合法的类型名。普通范围规则可运用于缩略前缀。也就是说,前缀所定义的元素只在元素中有效。
Namespace派什么用?namespace的作用是要避免命名冲突。如果我建立一项Web Service,其中的WSDL文件包含一个名为”foo”的元素,而你想要使用我的服务与另一项服务连接作为补充,这样的话另一项服务的WSDL文件就不能包含名为”foo”的元素。两个服务器程序只有在它们在两个事例中表示完全相同的东西时,才可以取相同的名字。如果有了表示区别的namespace,我的网络服务里的”foo”就可以表示完全不同于另一个网络服务里”foo”的含义。在你的客户端里,你只要加以限制就可以引用我的”foo”。
见下例:http://www.infotects.com/fooService#foo 就是完全限制的名字,相当于”carlos:foo”,如果我声明了carlos作为http://www.infotects.com/fooService的快捷方式。请注意namespace中的URL是用来确定它们的唯一性的,同时也便于定位。URL所指向的地方不必是实际存在的网络地址,也可以使用GUID来代替或补充URL。例如,GUID”335DB901-D44A-11D4-A96E-0080AD76435D”就是一个合法的namespace指派。
targetNamespace属性声明了一个namespace,元素中所有的声明的名字都列于其内。在WSDL示例中,<definitions>的targetNamespace 是http://tempuri.org/wsdl。这意味着所有在WSDL文档中声明的名字都属于这个namespace。<schema>元素有自己的targetNamespace属性,其值为 http://tempuri.org/xsd ,在<schma>元素中定义的所有名字都属于这个namespace而不是main的target namespace。
<schema>元素的以下这行声明了默认的namespace。Schema中所有有效的名字都属于这个namespace。
xmlns=“http://www.w3.org/2001/XMLSchema”
四、SOAP消息
对于使用WSDL的客户机和服务机来说,研究WSDL文件的一种方法就是决定什么来接受所发送的信息。尽管SOAP使用底层协议,如IP和HTTP等,但应用程序决定了服务器与客户机之间交互的高级协议。也就是说,进行一项操作,比如”echoint”把输入的整数送回,参数的数目、每个参数的类型、以及参数如何传送等因素决定了应用程序特定的协议。有很多方法可以确定此类协议,但我相信最好的方法就是使用WSDL。如果我们用这种视角来看待它,WSDL不只是一种接口协议,而且是一种协议特定的语言。它就是我们超越”固定”协议(IP、HTTP等)所需要的应用程序特定协议。
WSDL可以确定SOAP消息是否遵从RPC或文档风格。RPC风格的消息(就是示例中所用的)看起来像是函数调用。而文档风格的消息则更普通,嵌套层次更小。下面的XML消息就是示例WSDL文件解析后的发送/接受效果,解析使用的是MS SOAP Toolkit 2.0(MSTK2)中的SoapClient对象。
从客户端调用”foo(5131953)”函数:
<?xml version=“1.0” encoding=“UTF-8” standalone=“no”?>
<SOAP-ENV:Envelope
SOAP-ENV:encodingStyle=“http://schemas.xmlsoap.org/soap/encoding/”
xmlns:SOAP-ENV=“http://schemas.xmlsoap.org/soap/envelope/”>
<SOAP-ENV:Body>
<m:foo xmlns:m=“http://tempuri.org/message/”>
<arg>5131953</arg>
</m:foo>
</SOAP-ENV:Body>
</SOAP-ENV:Envelope>
从服务器接受的信息:
<?xml version=“1.0” encoding=“UTF-8” standalone=“no”?>
<SOAP-ENV:Envelope
SOAP-ENV:encodingStyle=“http://schemas.xmlsoap.org/soap/encoding/”
xmlns:SOAP-ENV=“http://schemas.xmlsoap.org/soap/envelope/”>
<SOAP-ENV:Body>
<SOAPSDK1:fooResponse xmlns:SOAPSDK1=“http://tempuri.org/message/”>
<result>5131953</result>
</SOAPSDK1:fooResponse>
</SOAP-ENV:Body>
</SOAP-ENV:Envelope>
两函数都调用了消息,其回应是有效的XML。SOAP消息由几部分组成,首先是<Envelop>元素,包含一个可选的<Header>元素以及至少一个<body>元素。Rpc函数所调用的消息体有一个根据操作”foo”命名的元素,而回应信息体有一个”fooResponse”元素。Foo元素有一个部分<arg>,就和WSDL中描述的一样,是单参数的。fooResponse也相应的有一个<result>的部分。注意encodingStyle、envelope和message的namespace和WSDL Bindings栏中的预定义的一致,重复如下:
<binding name=“SimpleBinding” type=“wsdlns:SimplePortType”>
<stk:binding preferredEncoding=“UTF-8” />
<soap:binding style=“rpc”
transport=“http://schemas.xmlsoap.org/soap/http”/>
<operation name=“foo”>
<soap:operation
soapAction=“http://tempuri.org/action/Simple.foo”/>
<input>
<soap:body use=“encoded”
namespace=“http://tempuri.org/message/”
encodingStyle=
“http://schemas.xmlsoap.org/soap/encoding/” />
</input>
<output>
<soap:body use=“encoded”
namespace=“http://tempuri.org/message/”
encodingStyle=
“http://schemas.xmlsoap.org/soap/encoding/” />
</output>
</operation>
</binding>
WSDL的Types栏和Messages栏中的XML Schema
WSDL数据类型是基于”XML Schema: Datatypes”(XSD)的,现在已经被W3C推荐。这一文档共有三个版本(1999,2000/10,2001),因此必须在namespace属性的<definitions>元素中指明所使用的是哪一个版本。
xmlns:xsd=“http://www.w3.org/2001/XMLSchema”
在本文中,我将只考虑2001版本。WSDL标准的推荐者强烈建议使用2001版。
在本栏和以后各部分,需使用以下简缩或前缀
前缀 代表的Namespace 描述
Soapenc http://schemas.xmlsoap.org/soap/encoding SOAP 1.1 encoding
Wsdl http://schemas.xmlsoap.org/wsdl/soap WSDL 1.1
Xsd http://www.w3.org/2001/XMLSchema XML Schema
五、XSD基类型
下表是直接从MSTK2文档中取出的,列举了MSTK2所支持的所有XSD基类型。它也告诉在客户端或服务器端的WSDL读取程序如何把XSD类型映射到在VB、C++和IDL中相应的类型。
XSD (Soap)类型 变量类型 VB C++ IDL Comments
anyURI VT_BSTR String BSTR BSTR
base64Binary VT_ARRAY | VT_UI1 Byte() SAFEARRAY SAFEARRAY(unsigned char)
Boolean VT_BOOL Boolean VARIANT_BOOL VARIANT_BOOL
Byte VT_I2 Integer short short 转换时验证范围有效性
Date VT_DATE Date DATE DATE 时间设为 oo:oo:oo
DateTime VT_DATE Date DATE DATE
Double VT_R8 Double double double
Duration VT_BSTR String BSTR BSTR 不转换和生效
ENTITIES VT_BSTR String BSTR BSTR 不转换和生效
ENTITY VT_BSTR String BSTR BSTR 不转换和生效
Float VT_R4 Single float float
GDay VT_BSTR String BSTR BSTR 不转换和生效
GMonth VT_BSTR String BSTR BSTR 不转换和生效
GMonthDay VT_BSTR String BSTR BSTR 不转换和生效
GYear VT_BSTR String BSTR BSTR 不转换和生效
GYearMonth VT_BSTR String BSTR BSTR 不转换和生效
ID VT_BSTR String BSTR BSTR 不转换和生效
IDREF VT_BSTR String BSTR BSTR 不转换和生效
IDREFS VT_BSTR String BSTR BSTR 不转换和生效
Int VT_I4 Long long long
Integer VT_DECIMAL Variant DECIMAL DECIMAL 转换时范围生效
Language VT_BSTR String BSTR BSTR 不转换和生效
Long VT_DECIMAL Variant DECIMAL DECIMAL 转换时范围生效
Name VT_BSTR String BSTR BSTR 不转换和生效
NCName VT_BSTR String BSTR BSTR 不转换和生效
negativeInteger VT_DECIMAL Variant DECIMAL DECIMAL 转换时范围生效
NMTOKEN VT_BSTR String BSTR BSTR 不转换和生效
NMTOKENS VT_BSTR String BSTR BSTR 不转换和生效
nonNegativeIntege VT_DECIMAL Variant DECIMAL DECIMAL 转换时范围生效
nonPositiveInteger VT_DECIMAL Variant DECIMA DECIMAL 转换时范围生效
normalizedString VT_BSTR String BSTR BSTR
NOTATION VT_BSTR String BSTR BSTR 不转换和生效
Number VT_DECIMAL Variant DECIMAL DECIMAL
positiveInteger VT_DECIMAL Variant DECIMAL DECIMAL 转换时范围生效
Qname VT_BSTR String BSTR BSTR 不转换和生效
Short VT_I2 Integer short short
String VT_BSTR String BSTR BSTR
Time VT_DATE Date DATE DATE 日设为1899年12月30日
Token VT_BSTR String BSTR BSTR 不转换和生效
unsignedByte VT_UI1 Byte unsigned char unsigned char
UnsignedInt VT_DECIMAL Variant DECIMAL DECIMAL 转换时范围生效
unsignedLong VT_DECIMAL Variant DECIMAL DECIMAL 转换时范围生效
unsignedShort VT_UI4 Long Long Long 转换时范围生效
XSD定义了两套内建的数据类型:原始的和派生的。在下文中查阅内建数据类型的层次十分有益:
http://www.w3.org/TR/2001/PR-xmlschema-2-20010330
六、complex类型
XML schema允许complex类型的定义,就像C里是struct。例如,为了定义类似如下的C的struct类型:
typedef struct {
string firstName;
string lastName;
long ageInYears;
float weightInLbs;
float heightInInches;
} PERSON;
我们可以写XML schema:
<xsd:complexType name=“PERSON”>
<xsd:sequence>
<xsd:element name=“firstName” type=“xsd:string”/>
<xsd:element name=“lastName” type=“xsd:string”/>
<xsd:element name=“ageInYears” type=“xsd:int”/>
<xsd:element name=“weightInLbs” type=“xsd:float”/>
<xsd:element name=“heightInInches” type=“xsd:float”/>
</xsd:sequence>
</xsd:complexType>
不过,complex类型可以表达比struct更多的信息。除了<sequence>以外,它还可以有其他的子元素,比如<all>
<xsd:complexType name=“PERSON”>
<xsd:all>
<xsd:element name=“firstName” type=“xsd:string”/>
<xsd:element name=“lastName” type=“xsd:string”/>
<xsd:element name=“ageInYears” type=“xsd:int”/>
<xsd:element name=“weightInLbs” type=“xsd:float”/>
<xsd:element name=“heightInInches” type=“xsd:float”/>
</xsd:all>
</xsd:complexType>
这意味着<element>的成员变量可以以任何顺序排列,每一个都是可选的。这和C中的struct类型不太一样。
注意内建数据类型string, int, float。C的string也是XML的string,float也类似。但C中的long类型在XML中是int(上表中)。
在WSDL文件中,像上面的complex类型可以在Types栏声明。例如,我可以用以下方式声明PERSON类型并用在Messages栏。
<?xml version=“1.0” encoding=“UTF-8” ?>
<definitions name=“FooSample”
targetNamespace=“http://tempuri.org/wsdl/”
xmlns:wsdlns=“http://tempuri.org/wsdl/”
xmlns:typens=“http://tempuri.org/xsd”
xmlns:xsd=“http://www.w3.org/2001/XMLSchema”
xmlns:soap=“http://schemas.xmlsoap.org/wsdl/soap/”
xmlns:stk=“http://schemas.microsoft.com/soap-toolkit/wsdl-extension”
xmlns=“http://schemas.xmlsoap.org/wsdl/”>
<types>
<schema targetNamespace=“someNamespace”
xmlns:typens=“someNamespace” >
<xsd:complexType name=“PERSON”>
<xsd:sequence>
<xsd:element name=“firstName” type=“xsd:string”/>
<xsd:element name=“lastName” type=“xsd:string”/>
<xsd:element name=“ageInYears” type=“xsd:int”/>
<xsd:element name=“weightInLbs” type=“xsd:float”/>
<xsd:element name=“heightInInches” type=“xsd:float”/>
</xsd:sequence>
</xsd:complexType>
</schema>
</types>
<message name=“addPerson”>
<part name=“person” type=“typens:PERSON”/>
</message>
<message name=“addPersonResponse”>
<part name=“result” type=“xsd:int”/>
</message>
</definitions>
上例中第一个消息由”addPerson”,并且有一个<part>,其类型为”PERSON”。PERSON类型是在Types栏声明的。
如果我们使用完整的WSDL文件包含以上的部分,并以之初始化MSTK2 SoapClient,它将成功的解析该文件。当然,它不会去调用<addPerson>。这是因为SoapClient本身并不知道如何处理complex类型,它需要定制类型映射来处理complex类型。MSTK2文档中有包含定制类型映射的示例。
还有另一种方法可以把<part>元素联系到类型声明。这就是使用元素。下例中我将Types栏中声明两个元素(”Person”和”Gendr”),然后我将在”addPerson”<message>中使用元素属性来引用它们。
<?xml version=“1.0” encoding=“UTF-8” ?>
<definitions name=“FooSample”
targetNamespace=“http://tempuri.org/wsdl/”
xmlns:wsdlns=“http://tempuri.org/wsdl/”
xmlns:typens=“http://tempuri.org/xsd”
xmlns:xsd=“http://www.w3.org/2001/XMLSchema”
xmlns:soap=“http://schemas.xmlsoap.org/wsdl/soap/”
xmlns:stk=“http://schemas.microsoft.com/soap-toolkit/wsdl-extension”
xmlns=“http://schemas.xmlsoap.org/wsdl/”>
<types>
<schema targetNamespace=“someNamespace”
xmlns:typens=“someNamespace” >
<element name=“Person”>
<xsd:complexType>
<xsd:sequence>
<xsd:element name=“firstName” type=“xsd:string”/>
<xsd:element name=“lastName” type=“xsd:string”/>
<xsd:element name=“ageInYears” type=“xsd:int”/>
<xsd:element name=“weightInLbs” type=“xsd:float”/>
<xsd:element name=“heightInInches” type=“xsd:float”/>
</xsd:sequence>
</xsd:complexType>
</element>
<element name=“Gender”>
<xsd:simpleType>
<xsd:restriction base=“xsd:string”>
<xsd:enumeration value=“Male” />
<xsd:enumeration value=“Female” />
</xsd:restriction>
</xsd:simpleType>
</element>
</schema>
</types>
<message name=“addPerson”>
<part name=“who” element=“typens:Person”/>
<part name=“sex” element=“typens:Gender”/>
</message>
<message name=“addPersonResponse”>
<part name=“result” type=“xsd:int”/>
</message>
</definitions>
Types栏中的Gender<element>里嵌入了枚举类型,其枚举值为”Male”“Female”。然后我又在”addPerson”<message>中通过元素属性而不是类型属性来引用它。
”元素属性”和”类型属性”在把某特定类型关联到<part>时有什么不同呢?使用元素属性,我们可以描述一个部分,它可以假定几个类型(就像变量一样),而是用类型属性我们就无法这样做。下例说明了这一点。
<?xml version=“1.0” encoding=“UTF-8” ?>
<definitions name=“FooSample”
targetNamespace=“http://tempuri.org/wsdl/”
xmlns:wsdlns=“http://tempuri.org/wsdl/”
xmlns:typens=“http://tempuri.org/xsd”
xmlns:xsd=“http://www.w3.org/2001/XMLSchema”
xmlns:soap=“http://schemas.xmlsoap.org/wsdl/soap/”
xmlns:stk=“http://schemas.microsoft.com/soap-toolkit/wsdl-extension”
xmlns=“http://schemas.xmlsoap.org/wsdl/”>
<types>
<schema targetNamespace=“someNamespace” xmlns:typens=“someNamespace”>
<xsd:complexType name=“PERSON”>
<xsd:sequence>
<xsd:element name=“firstName” type=“xsd:string”/>
<xsd:element name=“lastName” type=“xsd:string”/>
<xsd:element name=“ageInYears” type=“xsd:int”/>
<xsd:element name=“weightInLbs” type=“xsd:float”/>
<xsd:element name=“heightInInches” type=“xsd:float”/>
</xsd:sequence>
</xsd:complexType>
<xsd:complexType name=“femalePerson”>
<xsd:complexContent>
<xsd:extension base=“typens:PERSON” >
<xsd:element name=“favoriteLipstick” type=“xsd:string” />
</xsd:extension>
</xsd:complexContent>
</xsd:complexType>
<xsd:complexType name=“malePerson”>
<xsd:complexContent>
<xsd:extension base=“typens:PERSON” >
<xsd:element name=“favoriteShavingLotion” type=“xsd:string” />
</xsd:extension>
</xsd:complexContent>
</xsd:complexType>
<xsd:complexType name=“maleOrFemalePerson”>
<xsd:choice>
<xsd:element name=“fArg” type=“typens:femalePerson” >
<xsd:element name=“mArg” type=“typens:malePerson” />
</xsd:choice>
</xsd:complexType>
</schema>
</types>
<message name=“addPerson”>
<part name=“person” type=“typens:maleOrFemalePerson”/>
</message>
<message name=“addPersonResponse”>
<part name=“result” type=“xsd:int”/>
</message>
</definitions>
上例也告诉我们extension的派生。”femailPerson”和”malePerson”都是从”PERSON”派生出来的。它们各有一些额外的元素:”femalePerson”有”favoriteLipstick”元素,”malePerson”有”favoriteShavingLotion”元素。两派生类型都归入一个complex类型”maleOrFemalePerson”,使用的是<choice>构造。最后,在”adperson”<message>中,新类型有”person”<part>引用。这样,参数或<part>就可以是”femalePerson”或”malePerson”了。
像Internet协议之类的标准有没有为权威所利用,或者人们这样看待它是因为顺之所获的好处远远超出了代价?曾经有许多试图建立的标准都流产了。有时候,那些还没有普遍使用的标准甚至由法令或政府规定强行推出:Ada语言就是一例。
我相信正是跟随标准所带来的好处使它广泛接受。例如,对于铁路服务来说,真正重要的是,不同公司所铺设的铁路结合到一起,或者是来自好几个公司的产品协调的工作在一起。几家大的企业合力建立了SOAP标准。Web Service描述语言(WSDL)向这种Web Service的提供商和用户推出了方便的协调工作的方法,使我们能更容易的获得SOAP的种种好处。几家公司的铁道并在一起不算什么难事,他们所需遵循的只是两轨间的标准距离。对Web Service来说,这要复杂得多。我们必须先制定出指定接口的标准格式。
曾经有人说SOAP并不真需要什么接口描述语言。如果SOAP是交流纯内容的标准,那就需要一种语言来描述内容。SOAP消息确实带有某些类型信息,因此SOAP允许动态的决定类型。但不知道一个函数的函数名、参数的个数和各自类型,怎么可能去调用这个函数呢?没有WSDL,我可以从必备文档中确定调用语法,或者检查消息。随便何种方法,都必须有人参与,这个过程可能会有错。而使用了WSDL,我就可以通过这种跨平台和跨语言的方法使Web Service代理的产生自动化。就像COM和CORBA的IDL文件,WSDL文件由客户和服务器约定。
注意由于WSDL设计成可以绑定除SOAP以外的其他协议,这里我们主要关注WSDL在HTTP上和SOAP的关系。同样,由于SOAP目前主要用来调用远程的过程和函数,WSDL支持SOAP传输的文档规范。WSDL 1.1已经作为记录递交给W3C(见http://www.w3.org/TR/wsdl.html)
二、WSDL文档结构
若要理解XML文档,将之看作块状图表非常有用。下图以XML的文档形式说明了WSDL的结构,它揭示了WSDL文档五个栏之间的关系。
WSDL文档可以分为两部分。顶部分由抽象定义组成,而底部分则由具体描述组成。抽象部分以独立于平台和语言的方式定义SOAP消息,它们并不包含任何随机器或语言而变的元素。这就定义了一系列服务,截然不同的网站都可以实现。随网站而异的东西如序列化便归入底部分,因为它包含具体的定义。
l 抽象定义
Types
独立与机器和语言的类型定义
Messages
包括函数参数(输入与输出分开)或文档描述
PortTypes
引用消息部分中消息定义来描述函数签名(操作名、输入参数、输出参数)
2 具体描述
Bindings
PortTypes部分的每一操作在此绑定实现
Services
确定每一绑定的端口地址
下面的图中,箭头连接符代表文档不同栏之间的关系。点和箭头代表了引用或使用关系。双箭头代表"修改"关系。3-D的箭头代表了包含关系。这样,各Messages栏使用Types栏的定义,PortTypes栏使用Messages栏的定义;Bindings栏引用了PortTypes栏,Services栏引用Bindings栏,PortTypes和Bindings栏包含了operation元素,而Services栏包含了port元素。PortTypes栏里的operation元素由Bindings栏里的operation元素进一步修改或描述。
在此背景中,我将使用标准的XML术语来描述WSDL文档。Element是指XML的元素,而"attribute"指元素的属性。于是:
<element attribute="attribute-value">contents</element>
内容也可能由一个或多个元素以递归的方式组成。根元素是所有元素之中最高级的元素。子元素总是从属于另一个元素,父元素。
注意,文档之中可能只有一个Types栏,或根本没有。所有其他的栏可以只有零元素、单元素或是多元素。WSDL的列表要求所有的栏以固定的顺序出现:import, types, message, portType, binding, service。所有的抽象可以是单独存在于别的文件中,也可以从主文档中导入。
图一:抽象定义和具体定义
三、WSDL文件示例
让我们来研究一下WSDL文件,看看它的结构,以及如何工作。请注意这是一个非常简单的WSDL文档实例。我们的意图只是说明它最显著的特征。以下的内容中包括更加详细的讨论。
<?xml version=“1.0” encoding=“UTF-8” ?>
<definitions name=“FooSample”
targetNamespace=“http://tempuri.org/wsdl/”
xmlns:wsdlns=“http://tempuri.org/wsdl/”
xmlns:typens=“http://tempuri.org/xsd”
xmlns:xsd=“http://www.w3.org/2001/XMLSchema”
xmlns:soap=“http://schemas.xmlsoap.org/wsdl/soap/”
xmlns:stk=“http://schemas.microsoft.com/soap-toolkit/wsdl-extension”
xmlns=“http://schemas.xmlsoap.org/wsdl/”>
<types>
<schema targetNamespace=“http://tempuri.org/xsd”
xmlns=“http://www.w3.org/2001/XMLSchema”
xmlns:SOAP-ENC=“http://schemas.xmlsoap.org/soap/encoding/”
xmlns:wsdl=“http://schemas.xmlsoap.org/wsdl/”
elementFormDefault=“qualified” >
</schema>
</types>
<message name=“Simple.foo”>
<part name=“arg” type=“xsd:int”/>
</message>
<message name=“Simple.fooResponse”>
<part name=“result” type=“xsd:int”/>
</message>
<portType name=“SimplePortType”>
<operation name=“foo” parameterOrder=“arg” >
<input message=“wsdlns:Simple.foo”/>
<output message=“wsdlns:Simple.fooResponse”/>
</operation>
</portType>
<binding name=“SimpleBinding” type=“wsdlns:SimplePortType”>
<stk:binding preferredEncoding=“UTF-8” />
<soap:binding style=“rpc”
transport=“http://schemas.xmlsoap.org/soap/http”/>
<operation name=“foo”>
<soap:operation soapAction=“http://tempuri.org/action/Simple.foo”/>
<input>
<soap:body use=“encoded” namespace=“http://tempuri.org/message/”
encodingStyle=“http://schemas.xmlsoap.org/soap/encoding/” />
</input>
<output>
<soap:body use=“encoded” namespace=“http://tempuri.org/message/”
encodingStyle=“http://schemas.xmlsoap.org/soap/encoding/” />
</output>
</operation>
</binding>
<service name=“FOOSAMPLEService”>
<port name=“SimplePort” binding=“wsdlns:SimpleBinding”>
<soap:address location=“http://carlos:8080/FooSample/FooSample.asp”/>
</port>
</service>
</definitions>
以下是该实例文档的总述:稍后我将详细讨论每一部分的细节。
第一行申明该文档是XML。尽管这并不是必需的,但它有助于XML解析器决定是否解析WSDL文件或只是报错。第二行是WSDL文档的根元素:<definitions>。一些属性附属于根元素,就像<schema>子元素对于<types>元素。
<types>元素包含了Types栏。如果没有需要声明的数据类型,这栏可以缺省。在WSDL范例中,没有应用程序特定的types声明,但我仍然使用了Types栏,只是为了声明schema namespaces。
<message>元素包含了Messages栏。如果我们把操作看作函数,<message>元素定义了那个函数的参数。<message>元素中的每个<part>子元素都和某个参数相符。输入参数在<message>元素中定义,与输出参数相隔离--输出参数有自己的<message>元素。兼作输入、输出的参数在输入输出的<message>元素中有它们相应的<part>元素。输出<message>元素以”Response”结尾,就像以前所用的”fooResponse”。每个<part>元素都有名字和类型属性,就像函数的参数有参数名和参数类型。
用于交换文档时,WSDL允许使用<message>元素来描述交换的文档。
<part>元素的类型可以是XSD基类型,也可以是SOAP定义类型(soapenc)、WSDL定义类型(wsdl)或是Types栏定义的类型。
一个PortTypes栏中,可以有零个、单个或多个<portType>元素。由于抽象PortType定义可以放置在分开的文件中,在某个WSDL文件中没有<portType>元素是可能的。上面的例子里只是用了一个<portType>元素。而一个<portType>元素可在<operation>元素中定义一个或是多个操作。示例仅使用了一个名为”foo”的<operation>元素。这和某个函数名相同。<operation>元素可以有一个、两个、三个子元素:<input>, <output> 和<fault>元素。每个<input>和<output>元素中的消息都引用Message栏中的相关的<message>元素。这样,示例中的整个<portType>元素就和以下的C函数等效:
int foo(int arg);
这个例子足见XML和C相比要冗长的多。(包括<message>元素,XML在示例中共使用了12行代码来表达相同的单行函数声明。)
Bindings栏可以有零个、一个或者多个<binding>元素。它的意图是制定每个<operation>通过网络调用和回应。Services栏同样可以有零个、一个、多个<service>元素。它还包含了<port>元素,每个<port>元素引用一个Bindings栏里的<binding>元素。Bindings和Services栏都包含WSDL文档。
Namespace
<definitions>和子节点<schema>都是namespace属性:
<definitions name=“FooSample”
targetNamespace=“http://tempuri.org/wsdl/”
xmlns:wsdlns=“http://tempuri.org/wsdl/”
xmlns:typens=“http://tempuri.org/xsd”
xmlns:xsd=“http://www.w3.org/2001/XMLSchema”
xmlns:soap=“http://schemas.xmlsoap.org/wsdl/soap/”
xmlns:stk=“http://schemas.microsoft.com/soap-toolkit/wsdl-extension”
xmlns=“http://schemas.xmlsoap.org/wsdl/”>
<types>
<schema targetNamespace=“http://tempuri.org/xsd”
xmlns=“http://www.w3.org/2001/XMLSchema”
xmlns:SOAP-ENC=“http://schemas.xmlsoap.org/soap/encoding/”
xmlns:wsdl=“http://schemas.xmlsoap.org/wsdl/”
elementFormDefault=“qualified” >
</schema>
</types>
每个namespace属性都声明了一个缩略语,用在文档中。例如”xmlns:xsd”就为 http://www.w3.org/2001/XMLSchema定义了一个缩略语(xsd)。这就允许对该namespace的引用只需简单的在名字前加上前缀就可以了,如:”xsd:int”中的”xsd”就是合法的类型名。普通范围规则可运用于缩略前缀。也就是说,前缀所定义的元素只在元素中有效。
Namespace派什么用?namespace的作用是要避免命名冲突。如果我建立一项Web Service,其中的WSDL文件包含一个名为”foo”的元素,而你想要使用我的服务与另一项服务连接作为补充,这样的话另一项服务的WSDL文件就不能包含名为”foo”的元素。两个服务器程序只有在它们在两个事例中表示完全相同的东西时,才可以取相同的名字。如果有了表示区别的namespace,我的网络服务里的”foo”就可以表示完全不同于另一个网络服务里”foo”的含义。在你的客户端里,你只要加以限制就可以引用我的”foo”。
见下例:http://www.infotects.com/fooService#foo 就是完全限制的名字,相当于”carlos:foo”,如果我声明了carlos作为http://www.infotects.com/fooService的快捷方式。请注意namespace中的URL是用来确定它们的唯一性的,同时也便于定位。URL所指向的地方不必是实际存在的网络地址,也可以使用GUID来代替或补充URL。例如,GUID”335DB901-D44A-11D4-A96E-0080AD76435D”就是一个合法的namespace指派。
targetNamespace属性声明了一个namespace,元素中所有的声明的名字都列于其内。在WSDL示例中,<definitions>的targetNamespace 是http://tempuri.org/wsdl。这意味着所有在WSDL文档中声明的名字都属于这个namespace。<schema>元素有自己的targetNamespace属性,其值为 http://tempuri.org/xsd ,在<schma>元素中定义的所有名字都属于这个namespace而不是main的target namespace。
<schema>元素的以下这行声明了默认的namespace。Schema中所有有效的名字都属于这个namespace。
xmlns=“http://www.w3.org/2001/XMLSchema”
四、SOAP消息
对于使用WSDL的客户机和服务机来说,研究WSDL文件的一种方法就是决定什么来接受所发送的信息。尽管SOAP使用底层协议,如IP和HTTP等,但应用程序决定了服务器与客户机之间交互的高级协议。也就是说,进行一项操作,比如”echoint”把输入的整数送回,参数的数目、每个参数的类型、以及参数如何传送等因素决定了应用程序特定的协议。有很多方法可以确定此类协议,但我相信最好的方法就是使用WSDL。如果我们用这种视角来看待它,WSDL不只是一种接口协议,而且是一种协议特定的语言。它就是我们超越”固定”协议(IP、HTTP等)所需要的应用程序特定协议。
WSDL可以确定SOAP消息是否遵从RPC或文档风格。RPC风格的消息(就是示例中所用的)看起来像是函数调用。而文档风格的消息则更普通,嵌套层次更小。下面的XML消息就是示例WSDL文件解析后的发送/接受效果,解析使用的是MS SOAP Toolkit 2.0(MSTK2)中的SoapClient对象。
从客户端调用”foo(5131953)”函数:
<?xml version=“1.0” encoding=“UTF-8” standalone=“no”?>
<SOAP-ENV:Envelope
SOAP-ENV:encodingStyle=“http://schemas.xmlsoap.org/soap/encoding/”
xmlns:SOAP-ENV=“http://schemas.xmlsoap.org/soap/envelope/”>
<SOAP-ENV:Body>
<m:foo xmlns:m=“http://tempuri.org/message/”>
<arg>5131953</arg>
</m:foo>
</SOAP-ENV:Body>
</SOAP-ENV:Envelope>
从服务器接受的信息:
<?xml version=“1.0” encoding=“UTF-8” standalone=“no”?>
<SOAP-ENV:Envelope
SOAP-ENV:encodingStyle=“http://schemas.xmlsoap.org/soap/encoding/”
xmlns:SOAP-ENV=“http://schemas.xmlsoap.org/soap/envelope/”>
<SOAP-ENV:Body>
<SOAPSDK1:fooResponse xmlns:SOAPSDK1=“http://tempuri.org/message/”>
<result>5131953</result>
</SOAPSDK1:fooResponse>
</SOAP-ENV:Body>
</SOAP-ENV:Envelope>
两函数都调用了消息,其回应是有效的XML。SOAP消息由几部分组成,首先是<Envelop>元素,包含一个可选的<Header>元素以及至少一个<body>元素。Rpc函数所调用的消息体有一个根据操作”foo”命名的元素,而回应信息体有一个”fooResponse”元素。Foo元素有一个部分<arg>,就和WSDL中描述的一样,是单参数的。fooResponse也相应的有一个<result>的部分。注意encodingStyle、envelope和message的namespace和WSDL Bindings栏中的预定义的一致,重复如下:
<binding name=“SimpleBinding” type=“wsdlns:SimplePortType”>
<stk:binding preferredEncoding=“UTF-8” />
<soap:binding style=“rpc”
transport=“http://schemas.xmlsoap.org/soap/http”/>
<operation name=“foo”>
<soap:operation
soapAction=“http://tempuri.org/action/Simple.foo”/>
<input>
<soap:body use=“encoded”
namespace=“http://tempuri.org/message/”
encodingStyle=
“http://schemas.xmlsoap.org/soap/encoding/” />
</input>
<output>
<soap:body use=“encoded”
namespace=“http://tempuri.org/message/”
encodingStyle=
“http://schemas.xmlsoap.org/soap/encoding/” />
</output>
</operation>
</binding>
WSDL的Types栏和Messages栏中的XML Schema
WSDL数据类型是基于”XML Schema: Datatypes”(XSD)的,现在已经被W3C推荐。这一文档共有三个版本(1999,2000/10,2001),因此必须在namespace属性的<definitions>元素中指明所使用的是哪一个版本。
xmlns:xsd=“http://www.w3.org/2001/XMLSchema”
在本文中,我将只考虑2001版本。WSDL标准的推荐者强烈建议使用2001版。
在本栏和以后各部分,需使用以下简缩或前缀
前缀 代表的Namespace 描述
Soapenc http://schemas.xmlsoap.org/soap/encoding SOAP 1.1 encoding
Wsdl http://schemas.xmlsoap.org/wsdl/soap WSDL 1.1
Xsd http://www.w3.org/2001/XMLSchema XML Schema
五、XSD基类型
下表是直接从MSTK2文档中取出的,列举了MSTK2所支持的所有XSD基类型。它也告诉在客户端或服务器端的WSDL读取程序如何把XSD类型映射到在VB、C++和IDL中相应的类型。
XSD (Soap)类型 变量类型 VB C++ IDL Comments
anyURI VT_BSTR String BSTR BSTR
base64Binary VT_ARRAY | VT_UI1 Byte() SAFEARRAY SAFEARRAY(unsigned char)
Boolean VT_BOOL Boolean VARIANT_BOOL VARIANT_BOOL
Byte VT_I2 Integer short short 转换时验证范围有效性
Date VT_DATE Date DATE DATE 时间设为 oo:oo:oo
DateTime VT_DATE Date DATE DATE
Double VT_R8 Double double double
Duration VT_BSTR String BSTR BSTR 不转换和生效
ENTITIES VT_BSTR String BSTR BSTR 不转换和生效
ENTITY VT_BSTR String BSTR BSTR 不转换和生效
Float VT_R4 Single float float
GDay VT_BSTR String BSTR BSTR 不转换和生效
GMonth VT_BSTR String BSTR BSTR 不转换和生效
GMonthDay VT_BSTR String BSTR BSTR 不转换和生效
GYear VT_BSTR String BSTR BSTR 不转换和生效
GYearMonth VT_BSTR String BSTR BSTR 不转换和生效
ID VT_BSTR String BSTR BSTR 不转换和生效
IDREF VT_BSTR String BSTR BSTR 不转换和生效
IDREFS VT_BSTR String BSTR BSTR 不转换和生效
Int VT_I4 Long long long
Integer VT_DECIMAL Variant DECIMAL DECIMAL 转换时范围生效
Language VT_BSTR String BSTR BSTR 不转换和生效
Long VT_DECIMAL Variant DECIMAL DECIMAL 转换时范围生效
Name VT_BSTR String BSTR BSTR 不转换和生效
NCName VT_BSTR String BSTR BSTR 不转换和生效
negativeInteger VT_DECIMAL Variant DECIMAL DECIMAL 转换时范围生效
NMTOKEN VT_BSTR String BSTR BSTR 不转换和生效
NMTOKENS VT_BSTR String BSTR BSTR 不转换和生效
nonNegativeIntege VT_DECIMAL Variant DECIMAL DECIMAL 转换时范围生效
nonPositiveInteger VT_DECIMAL Variant DECIMA DECIMAL 转换时范围生效
normalizedString VT_BSTR String BSTR BSTR
NOTATION VT_BSTR String BSTR BSTR 不转换和生效
Number VT_DECIMAL Variant DECIMAL DECIMAL
positiveInteger VT_DECIMAL Variant DECIMAL DECIMAL 转换时范围生效
Qname VT_BSTR String BSTR BSTR 不转换和生效
Short VT_I2 Integer short short
String VT_BSTR String BSTR BSTR
Time VT_DATE Date DATE DATE 日设为1899年12月30日
Token VT_BSTR String BSTR BSTR 不转换和生效
unsignedByte VT_UI1 Byte unsigned char unsigned char
UnsignedInt VT_DECIMAL Variant DECIMAL DECIMAL 转换时范围生效
unsignedLong VT_DECIMAL Variant DECIMAL DECIMAL 转换时范围生效
unsignedShort VT_UI4 Long Long Long 转换时范围生效
XSD定义了两套内建的数据类型:原始的和派生的。在下文中查阅内建数据类型的层次十分有益:
http://www.w3.org/TR/2001/PR-xmlschema-2-20010330
六、complex类型
XML schema允许complex类型的定义,就像C里是struct。例如,为了定义类似如下的C的struct类型:
typedef struct {
string firstName;
string lastName;
long ageInYears;
float weightInLbs;
float heightInInches;
} PERSON;
我们可以写XML schema:
<xsd:complexType name=“PERSON”>
<xsd:sequence>
<xsd:element name=“firstName” type=“xsd:string”/>
<xsd:element name=“lastName” type=“xsd:string”/>
<xsd:element name=“ageInYears” type=“xsd:int”/>
<xsd:element name=“weightInLbs” type=“xsd:float”/>
<xsd:element name=“heightInInches” type=“xsd:float”/>
</xsd:sequence>
</xsd:complexType>
不过,complex类型可以表达比struct更多的信息。除了<sequence>以外,它还可以有其他的子元素,比如<all>
<xsd:complexType name=“PERSON”>
<xsd:all>
<xsd:element name=“firstName” type=“xsd:string”/>
<xsd:element name=“lastName” type=“xsd:string”/>
<xsd:element name=“ageInYears” type=“xsd:int”/>
<xsd:element name=“weightInLbs” type=“xsd:float”/>
<xsd:element name=“heightInInches” type=“xsd:float”/>
</xsd:all>
</xsd:complexType>
这意味着<element>的成员变量可以以任何顺序排列,每一个都是可选的。这和C中的struct类型不太一样。
注意内建数据类型string, int, float。C的string也是XML的string,float也类似。但C中的long类型在XML中是int(上表中)。
在WSDL文件中,像上面的complex类型可以在Types栏声明。例如,我可以用以下方式声明PERSON类型并用在Messages栏。
<?xml version=“1.0” encoding=“UTF-8” ?>
<definitions name=“FooSample”
targetNamespace=“http://tempuri.org/wsdl/”
xmlns:wsdlns=“http://tempuri.org/wsdl/”
xmlns:typens=“http://tempuri.org/xsd”
xmlns:xsd=“http://www.w3.org/2001/XMLSchema”
xmlns:soap=“http://schemas.xmlsoap.org/wsdl/soap/”
xmlns:stk=“http://schemas.microsoft.com/soap-toolkit/wsdl-extension”
xmlns=“http://schemas.xmlsoap.org/wsdl/”>
<types>
<schema targetNamespace=“someNamespace”
xmlns:typens=“someNamespace” >
<xsd:complexType name=“PERSON”>
<xsd:sequence>
<xsd:element name=“firstName” type=“xsd:string”/>
<xsd:element name=“lastName” type=“xsd:string”/>
<xsd:element name=“ageInYears” type=“xsd:int”/>
<xsd:element name=“weightInLbs” type=“xsd:float”/>
<xsd:element name=“heightInInches” type=“xsd:float”/>
</xsd:sequence>
</xsd:complexType>
</schema>
</types>
<message name=“addPerson”>
<part name=“person” type=“typens:PERSON”/>
</message>
<message name=“addPersonResponse”>
<part name=“result” type=“xsd:int”/>
</message>
</definitions>
上例中第一个消息由”addPerson”,并且有一个<part>,其类型为”PERSON”。PERSON类型是在Types栏声明的。
如果我们使用完整的WSDL文件包含以上的部分,并以之初始化MSTK2 SoapClient,它将成功的解析该文件。当然,它不会去调用<addPerson>。这是因为SoapClient本身并不知道如何处理complex类型,它需要定制类型映射来处理complex类型。MSTK2文档中有包含定制类型映射的示例。
还有另一种方法可以把<part>元素联系到类型声明。这就是使用元素。下例中我将Types栏中声明两个元素(”Person”和”Gendr”),然后我将在”addPerson”<message>中使用元素属性来引用它们。
<?xml version=“1.0” encoding=“UTF-8” ?>
<definitions name=“FooSample”
targetNamespace=“http://tempuri.org/wsdl/”
xmlns:wsdlns=“http://tempuri.org/wsdl/”
xmlns:typens=“http://tempuri.org/xsd”
xmlns:xsd=“http://www.w3.org/2001/XMLSchema”
xmlns:soap=“http://schemas.xmlsoap.org/wsdl/soap/”
xmlns:stk=“http://schemas.microsoft.com/soap-toolkit/wsdl-extension”
xmlns=“http://schemas.xmlsoap.org/wsdl/”>
<types>
<schema targetNamespace=“someNamespace”
xmlns:typens=“someNamespace” >
<element name=“Person”>
<xsd:complexType>
<xsd:sequence>
<xsd:element name=“firstName” type=“xsd:string”/>
<xsd:element name=“lastName” type=“xsd:string”/>
<xsd:element name=“ageInYears” type=“xsd:int”/>
<xsd:element name=“weightInLbs” type=“xsd:float”/>
<xsd:element name=“heightInInches” type=“xsd:float”/>
</xsd:sequence>
</xsd:complexType>
</element>
<element name=“Gender”>
<xsd:simpleType>
<xsd:restriction base=“xsd:string”>
<xsd:enumeration value=“Male” />
<xsd:enumeration value=“Female” />
</xsd:restriction>
</xsd:simpleType>
</element>
</schema>
</types>
<message name=“addPerson”>
<part name=“who” element=“typens:Person”/>
<part name=“sex” element=“typens:Gender”/>
</message>
<message name=“addPersonResponse”>
<part name=“result” type=“xsd:int”/>
</message>
</definitions>
Types栏中的Gender<element>里嵌入了枚举类型,其枚举值为”Male”“Female”。然后我又在”addPerson”<message>中通过元素属性而不是类型属性来引用它。
”元素属性”和”类型属性”在把某特定类型关联到<part>时有什么不同呢?使用元素属性,我们可以描述一个部分,它可以假定几个类型(就像变量一样),而是用类型属性我们就无法这样做。下例说明了这一点。
<?xml version=“1.0” encoding=“UTF-8” ?>
<definitions name=“FooSample”
targetNamespace=“http://tempuri.org/wsdl/”
xmlns:wsdlns=“http://tempuri.org/wsdl/”
xmlns:typens=“http://tempuri.org/xsd”
xmlns:xsd=“http://www.w3.org/2001/XMLSchema”
xmlns:soap=“http://schemas.xmlsoap.org/wsdl/soap/”
xmlns:stk=“http://schemas.microsoft.com/soap-toolkit/wsdl-extension”
xmlns=“http://schemas.xmlsoap.org/wsdl/”>
<types>
<schema targetNamespace=“someNamespace” xmlns:typens=“someNamespace”>
<xsd:complexType name=“PERSON”>
<xsd:sequence>
<xsd:element name=“firstName” type=“xsd:string”/>
<xsd:element name=“lastName” type=“xsd:string”/>
<xsd:element name=“ageInYears” type=“xsd:int”/>
<xsd:element name=“weightInLbs” type=“xsd:float”/>
<xsd:element name=“heightInInches” type=“xsd:float”/>
</xsd:sequence>
</xsd:complexType>
<xsd:complexType name=“femalePerson”>
<xsd:complexContent>
<xsd:extension base=“typens:PERSON” >
<xsd:element name=“favoriteLipstick” type=“xsd:string” />
</xsd:extension>
</xsd:complexContent>
</xsd:complexType>
<xsd:complexType name=“malePerson”>
<xsd:complexContent>
<xsd:extension base=“typens:PERSON” >
<xsd:element name=“favoriteShavingLotion” type=“xsd:string” />
</xsd:extension>
</xsd:complexContent>
</xsd:complexType>
<xsd:complexType name=“maleOrFemalePerson”>
<xsd:choice>
<xsd:element name=“fArg” type=“typens:femalePerson” >
<xsd:element name=“mArg” type=“typens:malePerson” />
</xsd:choice>
</xsd:complexType>
</schema>
</types>
<message name=“addPerson”>
<part name=“person” type=“typens:maleOrFemalePerson”/>
</message>
<message name=“addPersonResponse”>
<part name=“result” type=“xsd:int”/>
</message>
</definitions>
上例也告诉我们extension的派生。”femailPerson”和”malePerson”都是从”PERSON”派生出来的。它们各有一些额外的元素:”femalePerson”有”favoriteLipstick”元素,”malePerson”有”favoriteShavingLotion”元素。两派生类型都归入一个complex类型”maleOrFemalePerson”,使用的是<choice>构造。最后,在”adperson”<message>中,新类型有”person”<part>引用。这样,参数或<part>就可以是”femalePerson”或”malePerson”了。
发表评论
-
Axis2 XFire CXF 比较
2010-08-16 10:30 4005CXF最新版本:2.2.2 开源服务框架,可以通过API ... -
配置XFireClientFactoryBean时 设置 lookupServiceOnStartup = false
2010-08-05 16:45 1153<bean id="xxxService& ... -
Web Service描述语言 WSDL 详解(二)
2009-11-04 13:44 1815七、数组 XSD提供<list>结构来声明一个 ... -
Java调用以WSDL形式发布的web service
2009-11-04 13:40 2063webservice的 发布一般都 ... -
Web Service
2009-11-03 10:31 984一.Axis简介 1. 什么是SOAP ... -
Web service总述
2009-11-03 10:26 685分布式应用程序和浏览 ...
相关推荐
Web Service描述语言 WSDL 详解
标题:"Web Service描述语言 WSDL 详解.pdf" 描述:"Web Service描述语言 WSDL 详解.pdf" 标签:"service web wsdl" 部分内容:Web Service描述语言(WSDL)向这种WebService的提供商和用户推出了方便的协调工作的...
**Web Service描述语言(WSDL)详解** Web Service描述语言(Web Service Description Language,简称WSDL)是一种XML格式,用于定义网络服务的接口。它允许服务提供者明确地描述服务的位置、使用的消息协议以及...
Web Service描述语言(WSDL,Web Service Description Language)是一种基于XML的规范,用于定义Web服务的接口和通信细节。WSDL文件是Web服务的核心组成部分,它允许服务提供者明确地描述服务的功能、如何访问这些...
Web Service 描述语言(WSDL,Web Service Description Language)是一种基于XML的语言,用于定义网络服务的接口。WSDL被设计用来让服务提供者和服务消费者能够有效地进行通信,它描述了服务的能力、通信细节以及...
WebService描述语言WSDL是Web Service的一种重要组件,它提供了一种基于XML的描述服务的接口和消息的方法。WSDL文档描述了一个Web Service的接口,它包括一系列的抽象定义和具体定义,抽象定义与具体实现分离,有助...
### Web Service描述语言 (WSDL) 详解 #### 一、引言 WebService描述语言 (WSDL) 是一种用于描述Web服务的XML格式。通过WSDL,开发人员能够明确地指出服务的功能、如何调用这些功能以及数据如何被传递。WSDL的引入...
Web Service 描述语言(WSDL)是用于定义网络服务接口的一种XML格式,它详细描述了服务如何被调用以及如何响应。WSDL是SOAP(Simple Object Access Protocol)和其他Web服务技术的重要组成部分,允许服务消费者和...