webservice入门day1

本文介绍了Web Service的基本概念,包括什么是远程调用技术和Web Service,详细讲解了WSDL、SOAP、UDDI的核心内容。通过一个天气查询服务的实例展示了服务端和客户端的开发过程,并探讨了Web Service的优缺点。此外,还讨论了四种不同的客户端调用方式,以及如何通过注解修改WSDL内容。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

1.什么是webservice

    1.1什么是远程调用技术

    远程调用技术是系统和系统之间的调用,例如支付宝与银行之间,调用天气服务,调用查询手机归属地服务,调用查询违章信息服务等等

    提供服务的称为服务端,调用该服务的是服务端,两者之间通过webservice调用

    1.2什么是webservice

    如图,webservice就是通过http请求传递SOAP协议数据的一种远程调用技术

   

    其中,请求和响应,通讯方式都是http请求中的post请求,数据解析也都采用的是遵循SOAP(规范xml标签)协议的xml

    WSDL:它是webservice服务端的一个使用说明书,描述接口,方法,参数和返回值

    UDDI:提供webservice服务端的注册和搜索功能.(开发中很少用,因为一般项目中需要的服务是明确的)

2.webservice入门程序
    java中有三种webservice规范,分别是JAX-WS(JAX-RPC)(小公司用的多)、JAXM&SAAJ(用的少)、JAX-RS(比较新,大公司在用)。

    服务端开发案例:

    需求:1.服务端:发布一个天气查询服务,接收客户端城市名,返回天气数据给客户端

            2.客户端 : 发送城市名称给服务端,接收服务端的返回天气数据,打印

    实现:

            1.服务端实现

            第一步:创建SEI(Service Endpoint Interface)接口,本质上就是Java接口(面向接口编程)

            第二步:创建SEI接口实现类

            

            第三步:发布服务

            注意这一步,是使用Endpointpublish方法来发布服务的,参数一个是发布地址,一个是实现类

            

            第四步:测试服务是否发布成功

            是通过阅读使用说明书,确定客户端调用的接口,方法,参数和返回值存在,证明服务发布成功

            如何阅读说明书?在浏览器输入发布地址,进入服务页,点击WSDL进入说明书

            说明书内部逻辑:

         

        2.客户端实现

        第一步:使用wsimport命令生成客户端代码

        第二步:根据使用说明书,在客户端类中使用客户端代码调用服务端

                //创建服务视图

                //获取服务实现类

                //调用查询方法,打印

        3.webservice优缺点

        优点:

l  发送方式采用http的post发送,http的默认端口是80,防火墙默认不拦截80,所以跨防火墙

l  采用XML格式封装数据,XML是跨平台的,所以webservice也可以跨平台。

l  Webservice支持面向对象

        缺点:

l  采用XML格式封装数据,所以在传输过程中,要传输额外的标签,随着SOAP协议的不断完善,标签越来越大,导致webservice性能下降

4.WSDL
   4.1定义

   WSDL及web服务描述语言,他是webservice服务端使用说明书,说明服务端接口、方法、参数和返回值,WSDL是随服务发布成功,自动生成,无需编写

   4.2文档结构

  

  •     <service> 服务视图,webservice的服务结点,它包括了服务端点
  •     <binding>     为每个服务端点定义消息格式和协议细节
  •     <portType>   服务端点,描述 web service可被执行的操作方法,以及相关的消息,通过binding指向portType
  •     <message>   定义一个操作(方法)的数据参数(可有多个参数)
  •     <types>        定义 web service 使用的全部数据类型

  4.3阅读方式:从下往上


5.SOAP
    5.1定义

    SOAP即简单对象访问协议,他是使用http发送的XML格式的数据,它可以跨平台,跨防火墙,SOAP不是webservice的专有协议。SOAP=http+xml

       5.2协议格式

  •   必须有 Envelope 元素,此元素将整个 XML 文档标识为一条 SOAP 消息
  •   可选的 Header 元素,包含头部信息
  •   必须有 Body 元素,包含所有的调用和响应信息
  •   可选的 Fault 元素,提供有关在处理此消息所发生错误的信息

       5.3TCP/IP monitor

        代理原理:

      

          在preferences下搜索TCP/IP monitor,进行代理设置,如果代理端口可以正常访问,则说明代理成功

            

      5.4 SOAP1.1

      请求

POST /weather HTTP/1.1

Accept: text/xml, multipart/related

Content-Type: text/xml; charset=utf-8

SOAPAction: "http://ws.jaxws.ws.itcast.cn/WeatherInterfaceImpl/queryWeatherRequest"

User-Agent: JAX-WS RI 2.2.4-b01

Host: 127.0.0.1:54321

Connection: keep-alive

Content-Length: 214

 

<?xml version="1.0" ?>

<S:Envelope xmlns:S="http://schemas.xmlsoap.org/soap/envelope/">

<S:Body><ns2:queryWeather xmlns:ns2="http://ws.jaxws.ws.itcast.cn/"><arg0>北京</arg0></ns2:queryWeather>

</S:Body>

</S:Envelope>

    响应

HTTP/1.1 200 OK

Transfer-encoding: chunked

Content-type: text/xml; charset=utf-8

Date: Thu, 26 Nov 2015 03:14:29 GMT

 

<?xml version="1.0" ?>

<S:Envelope xmlns:S="http://schemas.xmlsoap.org/soap/envelope/">

<S:Body>

<ns2:queryWeatherResponse xmlns:ns2="http://ws.jaxws.ws.itcast.cn/"><return>晴</return></ns2:queryWeatherResponse>

</S:Body>

</S:Envelope>

      5.5 SOAP1.2

如何发布SOAP1.2服务端?

    Jaxws不支持SOAP1.2服务端发布,直接发布会报如下异常

    如果想发布SOAP1.2服务端,需要在服务端引入第三方JAR(jaxws-ri-2.2.8)

    在实现类上加入如下注解@BindingType(SOAPBinding.SOAP12HTTP_BINDING)

    请求:

POST /weather HTTP/1.1

Accept: application/soap+xml, multipart/related

Content-Type: application/soap+xml; charset=utf-8;

action="http://ws.jaxws.ws.itcast.cn/WeatherInterfaceImpl/queryWeatherRequest"

User-Agent: JAX-WS RI 2.2.4-b01

Host: 127.0.0.1:54321

Connection: keep-alive

Content-Length: 212

 

<?xml version="1.0" ?>

<S:Envelope xmlns:S="http://www.w3.org/2003/05/soap-envelope">

<S:Body><ns2:queryWeather xmlns:ns2="http://ws.jaxws.ws.itcast.cn/"><arg0>北京</arg0></ns2:queryWeather>

</S:Body>

</S:Envelope>

    响应:

HTTP/1.1 200 OK

Transfer-encoding: chunked

Content-type: application/soap+xml; charset=utf-8

Date: Thu, 26 Nov 2015 03:25:24 GMT

 

<?xml version='1.0' encoding='UTF-8'?>

<S:Envelope xmlns:S="http://www.w3.org/2003/05/soap-envelope">

<S:Body>

<ns2:queryWeatherResponse xmlns:ns2="http://ws.jaxws.ws.itcast.cn/"><return>晴</return></ns2:queryWeatherResponse>

</S:Body>

</S:Envelope>

    5.6 SOAP1.1和SOAP1.2区别

    相同点:

        (1)请求发送方式相同:都是POST

        (2)协议内容相同:都有Envelope和Body标签

    不同点:

        (1)数据格式不同:content-type不同       

            SOAP1.1:text/xml;charset=utf-8

            SOAP1.2:application/soap+xml;charset=utf-8

        (2)命名空间不同:

            SOAP1.1:http://schemas.xmlsoap.org/soap/envelope/

            SOAP1.2:http://www.w3.org/2003/05/soap-envelope

6.UDDI

    UDDI 是一种目录服务,企业可以使用它对 Web services 进行注册和搜索,UDDI并不像 WSDL SOAP 一样深入人心,因为很多时候,使用者知道 Web 服务的位置(通常位于公司的企业内部网中)

   

7.WebService的四种客户端调用方式

   公网服务地址:http://www.webxml.com.cn/zh_cn/index.aspx

   7.1 第一种生成客户端调用方式

       7.1.1 Wsimport命令介绍

       Wsimport就是jdk提供的的一个工具,他作用就是根据WSDL地址生成客户端代码

       Wsimport位置JAVA_HOME/bin

       Wsimport常用的参数:

            -s,生成java文件的

            -d,生成class文件的,默认的参数

            -p,指定包名的,如果不加该参数,默认包名就是wsdl文档中的命名空间的倒序

      Wsimport仅支持SOAP1.1客户端的生成

    7.1.2调用公网手机号归属地查询服务

       第一步:根据服务端wsdl使用wsimport生成客户端代码

wsimport -p cn.itcast.mobile -s . http://webservice.we

bxml.com.cn/WebServices/MobileCodeWS.asmx?wsdl

       第二步:阅读使用说明书,使用生成客户端代码调用服务端

    

package cn.itcast.mobile.client;

 

import cn.itcast.mobile.MobileCodeWS;

import cn.itcast.mobile.MobileCodeWSSoap;

 

/**

 *

 * <p>Title: MobileClient.java</p>

 * <p>Description:公网手机号查询客户端</p>

 * <p>Company: www.itcast.com</p>

 * @author  传智.at

 * @date    20151126日下午3:16:05

 * @version 1.0

 */

publicclass MobileClient {

 

   publicstaticvoid main(String[] args) {

      //获取服务,即找到项目------对应wsdl说明书里的service name

    

      MobileCodeWS mobileCodeWS = new MobileCodeWS();

      //获取服务实现类,即找到项目里的类----对应wsdl说明书里的portType

    

      MobileCodeWSSoap mobileCodeWSSoap = mobileCodeWS.getPort(MobileCodeWSSoap.class);

      //调用查询方法,即找到类里的方法------就是我们想要调用的方法,对应wsdl说明书里的getMobileCodeInfo


      String reuslt = mobileCodeWSSoap.getMobileCodeInfo("13888888", null);

      System.out.println(reuslt);

   }

}

 

    7.1.3公网天气服务端查询

package cn.itcast.mobile.client;

 

import java.util.List;

 

import cn.itcast.weather.ArrayOfString;

import cn.itcast.weather.WeatherWS;

import cn.itcast.weather.WeatherWSSoap;

 

/**

 *

 * <p>Title: WeatherClient.java</p>

 * <p>Description:公网天气查询客户端</p>

 * <p>Company: www.itcast.com</p>

 * @author  传智.at

 * @date    20151126日下午3:24:12

 * @version 1.0

 */

publicclass WeatherClient {

 

   publicstaticvoid main(String[] args) {

      WeatherWS weatherWS = new WeatherWS();

      WeatherWSSoap weatherWSSoap = weatherWS.getPort(WeatherWSSoap.class);

      ArrayOfString  arrayOfString = weatherWSSoap.getWeather("北京", "");

      List<String> list = arrayOfString.getString();

     

      for(String str : list){

         System.out.println(str);

      }

   }

}

 

    该种方式使用简单,但一些关键的元素在代码生成时写死到生成代码中,不方便维护,所以仅用于测试.

    7.2.Service编程调用方式

package cn.itcast.mobile.client;

 

import java.io.IOException;

import java.net.MalformedURLException;

import java.net.URL;

 

import javax.xml.namespace.QName;

import javax.xml.ws.Service;

 

import cn.itcast.mobile.MobileCodeWSSoap;

 

/**

 *

 * <p>Title: ServiceClient.java</p>

 * <p>Description:Service编程实现服务端调用</p>

 * <p>Company: www.itcast.com</p>

 * @author  传智.at

 * @date    20151126日下午3:43:55

 * @version 1.0

 */

publicclass ServiceClient {

 

   publicstaticvoid main(String[] args) throws IOException {

      //创建WSDLURL,注意不是服务地址

      URL url = new URL("http://webservice.webxml.com.cn/WebServices/MobileCodeWS.asmx?wsdl");

     

      //创建服务名称

      //1.namespaceURI - 命名空间地址

      //2.localPart - 服务视图名

      QName qname = new QName("http://WebXml.com.cn/", "MobileCodeWS");

     

      //创建服务视图

      //参数解释:

      //1.wsdlDocumentLocation - wsdl地址

      //2.serviceName - 服务名称

      Service service = Service.create(url, qname);

      //获取服务实现类

      MobileCodeWSSoap mobileCodeWSSoap = service.getPort(MobileCodeWSSoap.class);

      //调用查询方法

      String result = mobileCodeWSSoap.getMobileCodeInfo("1866666666", "");

      System.out.println(result);

   }

}

 

    该种方式可以自定义关键元素,方便以后维护,是一种标准的开发方式

    7.3.HttpURLConnection调用方式

    开发步骤:

    第一步:创建服务地址

    第二步:打开一个通向服务地址的连接

    第三步:设置参数

    设置POST,POST必须大写,如果不大写,报如下异常

    

    如果不设置输入输出,会报如下异常

    

    第四步:组织SOAP数据,发送请求

    第五步:接收服务端响应,打印

package cn.itcast.mobile.client;

 

import java.io.BufferedReader;

import java.io.IOException;

import java.io.InputStream;

import java.io.InputStreamReader;

import java.io.OutputStream;

import java.net.HttpURLConnection;

import java.net.MalformedURLException;

import java.net.URL;

 

/**

 *

 * <p>Title: HttpClient.java</p>

 * <p>Description:HttpURLConnection调用方式</p>

 * <p>Company: www.itcast.com</p>

 * @author  传智.at

 * @date    20151126日下午3:58:57

 * @version 1.0

 */

publicclass HttpClient {

 

   publicstaticvoid main(String[] args) throws IOException {

      //第一步:创建服务地址,不是WSDL地址

      URL url = new URL("http://webservice.webxml.com.cn/WebServices/MobileCodeWS.asmx");

      //第二步:打开一个通向服务地址的连接

      HttpURLConnection connection = (HttpURLConnection) url.openConnection();

      //第三步:设置参数

      //3.1发送方式设置:POST必须大写

      connection.setRequestMethod("POST");

      //3.2设置数据格式:content-type

      connection.setRequestProperty("content-type", "text/xml;charset=utf-8");

      //3.3设置输入输出,因为默认新创建的connection没有读写权限,

      connection.setDoInput(true);

      connection.setDoOutput(true);

 

      //第四步:组织SOAP数据,发送请求

      String soapXML = getXML("15226466316");

      OutputStream os = connection.getOutputStream();

      os.write(soapXML.getBytes());

      //第五步:接收服务端响应,打印

      intresponseCode = connection.getResponseCode();

      if(200 == responseCode){//表示服务端响应成功

         InputStream is = connection.getInputStream();

         InputStreamReader isr = new InputStreamReader(is);

         BufferedReader br = new BufferedReader(isr);

        

         StringBuilder sb = new StringBuilder();

         String temp = null;

         while(null != (temp = br.readLine())){

            sb.append(temp);

         }

         System.out.println(sb.toString());

        

         is.close();

         isr.close();

         br.close();

      }

 

      os.close();

   }

  

   /**

    * <?xml version="1.0" encoding="utf-8"?>

<soap:Envelope xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:soap="http://schemas.xmlsoap.org/soap/envelope/">

  <soap:Body>

    <getMobileCodeInfo xmlns="http://WebXml.com.cn/">

      <mobileCode>string</mobileCode>

      <userID>string</userID>

    </getMobileCodeInfo>

  </soap:Body>

</soap:Envelope>

    * @param phoneNum

    * @return

    */

   publicstatic String getXML(String phoneNum){

      String soapXML = "<?xml version=\"1.0\" encoding=\"utf-8\"?>"

      +"<soap:Envelope xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\" xmlns:xsd=\"http://www.w3.org/2001/XMLSchema\" xmlns:soap=\"http://schemas.xmlsoap.org/soap/envelope/\">"

         +"<soap:Body>"

          +"<getMobileCodeInfo xmlns=\"http://WebXml.com.cn/\">"

             +"<mobileCode>"+phoneNum+"</mobileCode>"

            +"<userID></userID>"

          +"</getMobileCodeInfo>"

        +"</soap:Body>"

      +"</soap:Envelope>";

      returnsoapXML;

   }

}

 

    7.4.Ajax调用方式

<!doctype html>

<html lang="en">

 <head>

  <meta charset="UTF-8">

  <title>Document</title>

  <script type="text/javascript">

         function queryMobile(){

                   //创建XMLHttpRequest对象

                   var xhr = new XMLHttpRequest();

                   //打开连接

                  xhr.open("post","http://webservice.webxml.com.cn/WebServices/MobileCodeWS.asmx",true);

                   //设置数据类型

                   xhr.setRequestHeader("content-type","text/xml;charset=utf-8");

                   //设置回调函数

                   xhr.onreadystatechange=function(){

                            //判断是否发送成功和判断服务端是否响应成功

                            if(4 == xhr.readyState && 200 == xhr.status){

                                     alert(xhr.responseText);

                            }

                   }

                   //组织SOAP协议数据

                   var soapXML = "<?xml version=\"1.0\" encoding=\"utf-8\"?>"

                   +"<soap:Envelope xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\" xmlns:xsd=\"http://www.w3.org/2001/XMLSchema\" xmlns:soap=\"http://schemas.xmlsoap.org/soap/envelope/\">"

                            +"<soap:Body>"

                       +"<getMobileCodeInfo xmlns=\"http://WebXml.com.cn/\">"

                                +"<mobileCode>"+document.getElementById("phoneNum").value+"</mobileCode>"

                         +"<userID></userID>"

                       +"</getMobileCodeInfo>"

                     +"</soap:Body>"

                   +"</soap:Envelope>";

                   alert(soapXML);

                   //发送数据

                   xhr.send(soapXML);

         }

  </script>

 </head>

 <body>

  手机号查询:<input type="text" id="phoneNum"/> <input type="button" value="查询" οnclick="javascript:queryMobile();"/>

 </body>

</html>

11.深入开发:用注解修改WSDL内容

WebService的注解都位于javax.jws包下:

@WebService-定义服务,在public class上边

targetNamespace:指定命名空间

name:portType的名称

portName:port的名称

serviceName:服务名称

endpointInterface:SEI接口地址,如果一个服务类实现了多个接口,只需要发布一个接口的方法,可通过此注解指定要发布服务的接口。

@WebMethod-定义方法,在公开方法上边

       operationName:方法名

       exclude:设置为true表示此方法不是webservice方法,反之则表示webservice方法,默认是false

@WebResult-定义返回值,在方法返回值前边

       name:返回结果值的名称

@WebParam-定义参数,在方法参数前边

       name:指定参数的名称

作用:

通过注解,可以更加形象的描述Web服务。对自动生成的wsdl文档进行修改,为使用者提供一个更加清晰的wsdl文档。

当修改了WebService注解之后,会影响客户端生成的代码。调用的方法名和参数名也发生了变化





评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值