WSDL原理

参考文章:https://blog.csdn.net/acmman/article/details/52179346

我们经常见到WebService中有“WSDL”字样,我们访问的WebService接口也是充满了XML配置,那么WSDL和XML有什么关系?我们的服务是怎么样让堆放获取的呢?

想要知道以上这些,我们要了解WSDL、SOAP和UDDI。

其实WSDL就是Web Service Definition Lanauage,即WebService定义语言。也就是我们之前的浏览器上访问出的xml,就是WSDL构造出的对外暴露的WebService定义语言,在客户端会通过wsdl2Java指令来翻译服务端提供的这段XML,从而在客户都获取WebService提供的方法。

举个webService例子:

<definitions targetNamespace="http://impl.ws.cxf.java.org/" name="HelloworldWs">
    <import namespace="http://ws.cxf.java.org/" location="http://127.0.0.1:8082/sayHi?wsdl=1"/>
    <binding name="HelloworldWsPortBinding" type="ns1:HelloWorld"><soap:binding transport="http://schemas.xmlsoap.org/soap/http" style="document"/>
        <operation name="sayHi">
            <soap:operation soapAction=""/>
            <input>
            <soap:body use="literal"/>
        </input>
        <output>
            <soap:body use="literal"/>
            </output>
    </operation>
    <operation name="getCatsByUser">
        <soap:operation soapAction=""/>
        <input>
            <soap:body use="literal"/>
        </input>
        <output>
        <soap:body use="literal"/>
        </output>
    </operation>
     </binding>
     <service name="HelloworldWs">
         <port name="HelloworldWsPort" binding="tns:HelloworldWsPortBinding">
             <soap:address location="http://127.0.0.1:8082/sayHi"/>
         </port>
     </service>
</definitions>

标签收起后:

<definitions>
     <import/>
     <binding>
    ......
     </binding>
     <service name="HelloworldWs">
        ......
     </service>
</definitions>

我们详细解析一下这段XML。

首先第一段标签名为definitions,targetNamespace相当于Java中的package(xmlns相当于Java的import)。其内容http://impl.ws.cxf.java.org/和我们服务端的实现类的包名一样。

import标签中的namespace即是我们的接口所在的包,location是WebService对外暴露的接口的WSDL文档路径(http://127.0.0.1:8082/sayHi?wsdl=1)。

所以上面<import/>是接口的定义,下面的<binding>和<service>标签是Service的实现。

所以一个WSDL文档大致分为以下模块:

1.接口部分文档

我们首先来分析接口部分文档,我们来访问一下刚刚上面import标签中location的路径http://127.0.0.1:8082/sayHi?wsdl=1:

收起里面复杂的数据,其实大致为以下结构

其中type存放的是一个标准schema格式的数据

<types>
<xsd:schema>
<xsd:import namespace="http://ws.cxf.java.org/" schemaLocation="http://127.0.0.1:8082/sayHi?xsd=1"/>
</xsd:schema>
</types>

xsd:import中引入了一个namespace,location中引入了该接口所有数据类型,我们来访问一下http://127.0.0.1:8082/sayHi?xsd=1,得到:

 

我们可以看到里面包含了所有方法的输入参数类型和返回参数类型,其中的minOccurs是该参数出现的最小次数,maxOccurs是参数出现的最大次数。

像<xs:element name="return" type="tns:cat" minOccurs="0" maxOccurs="unbounded"/>这个标签的意思就是,像List中的Cat这种返回类型可能出现的个数是0-N个。

可以看到wsdl文档的最后有一个portType的标签对,里面存放了所有可以调用的WebService服务,然后里面是服务的输入输出,其中的message属性就对应上面的message属性。

 其中message属性的详细定义,就在types的xsd:schema中定义,而其中的xsd:import标签就引入了各种格式的定义文档路径。

所以,一次WebService的调用,其实并不是方法调用,而是发送SOAP消息(即XML文档片段)。

对于sayHi操作来说,传入的消息是:

<sayHi>
    <arg0>字符串</arg0>
</sayHi>
传出的消息是:
<sayHiResponse>
    <return>字符串</return>
</sayHiResponse>

对于getCatsByUser操作来说,传入的消息是:

<getCatsByUser>
    <arg0>
        <id>整数值</id>
        <name>字符串</name>
        <pass>字符串</pass>
        <address>字符串</address>
    </arg0>
</getCatsByUser>
传出的消息是:
<getCatsByUserResponse>
    <return>   --可能出现0-N次
        <id>整数值</id>
        <name>字符串</name>
        <color>字符串</color>
    </return>
</getCatsByUserResponse>

2.实现类部分文档

我们回到之前的路径http://127.0.0.1:8082/sayHi?wsdl:

 上面我们访问的http://127.0.0.1:8082/sayHi?wsdl=1是接口类的定义文档,所以上面<import/>是接口的定义,下面的<binding>和<service>标签是Service的实现。

bingding中绑定了可操作的方法,以及输入输出的参数。

service中定义了该WebService对外提供的服务接口地址。

所以整个WSDL的文档结构就是:

 通俗的说,WSDL文档描述了WebService如下3个方面:

  • what:该WebService包含“什么”操作。
  • how:该WebService的操作应该“怎么”调用。
  • where:该WebService的服务地址。

调用一次WebService的本质:

  1. 客户端把调用方法参数,转换成XML文档片段(SOAP消息,input消息),该文档片段必须服务WSDL定义的格式。
  2. 通过网络,把XML文档片段从客户端传送给服务器。
  3. 服务器接收到XML文档片段。
  4. 服务器解析XML片段,提取其中的数据,并把数据转换成调用WebService所需要的参数值。
  5. 服务器执行方法。
  6. 把执行的方法得到的返回值,再次转换生成XML文档片段(SOAP,output消息),该文档片段必须符合WSDL定义的格式。
  7. 通过网络,把XML文档片段传送给客户端。
  8. 客户端接收到XML文档片段。
  9. 客户端解析XML片段,提取其中的数据,并把数据转换成调用WebService所需要的返回值。

从上面调用本质来看,要一个语言支持WebService,唯一的要求就是:该语言支持XML文档解析、生成、支持网络传输。

只要得到WebService文档,程序就可以调用WebService。

猜你喜欢

转载自www.cnblogs.com/susanhonly/p/11937151.html