前言:

随着近几年来,SOA,EAI等架构体系的日渐成熟,Webservice越来越炽手可热,尤其是在企业做异质平台整合时成为了首选的技术。

Java的Webservice技术更是层出不穷,比較流行的有:

Axis2,Spring WS以及Jaxws。

本人在日常工作和以往project中,在使用了上述这些Webservice后进行了总结,比較,终于认为jaxws是眼下最标准,须要额外第三方插件最少,配置最少最灵活的webservice。

JAXWS适合差点儿全部Webserviceclient的调用,因此不少巨头型的厂商如:IBM,Weblogic等,在他们的产品上都使用了以JAXWS为标准的Webservice接口。

本教程分成五天,为0基础教程。

通过本教程,能够使一个没有Webservice概念或者没有写过Webservice的JAVA Resource高速上手入门,并能满足一般中小型项目中Webservice的应用。

对于Webservice Security,在(0基础)教程中并不提供,会在高级教程中具体描写叙述。

只是真正利用Webservice Security特性即XML加密技术的project并不多,少之又少,大多还是以http: //xxx/xxxService?userId=&password=这种形式来进行“假安全”通讯的。

必经我们的大部分项目是执行在Intranet里的,并且有非常好的监控和布防。

以下開始我们的教程。

第一天

目标:

1. 理解jaxws

2. 写jaxws之前的准备工作

3. 一切始于HelloWorld

4. 理解同步,异步

一、理解jaxws

1.1JAX-WS概述

JAX-WS2.0 的全称为 Java API for XML-Based Webservices (JAX-WS) 2.0。JAX-WS 2.0 是对 JAX-RPC 1.0 规范的扩展,是 JAX-RPC 1.1 的兴许版本号, JAX-RPC 2.0 标准公布不久后便被又一次命名为 JAX-WS 2.0。 JAX-WS 2.0 是面向 Java 5 的开发 Web services 的最新编程标准,它提供了新的编程模型和对以往的 JAX-RPC 方式的 Web services 进行了增强。 JAX-WS2.0 (JSR 224)是Sun新的web services协议栈,是一个全然基于标准的实现。在binding层,使用的是the Java Architecture for XMLBinding (JAXB, JSR 222),在parsing层,使用的是the Streaming API for XML (StAX, JSR 173),同一时候它还全然支持schema规范。

1.2JAX-WS 2.1特性

支持SOAP 1.1(默认)、1.2

支持XML/HTTP Binding

支持WS-Addressing

支持document/literal样式

支持WS-I Basic Profile 1.1

支持消息传输优化机制(Message Transmission Optimization Mechanism,MTOM)

二、写jaxws之前的准备工作

2.1JDKjavaversion "1.6.0_x"。

2.2JAX-WS RI 2.1.1 in JDK 6

2.2.1 JAX-WS RI 2.1.1安装注解

JAX-WS RI组件下载后为一个”.jar”文件,它并不能直接在project中使用,它是一个以JAVA Swing为界面的JAXWS的安装程序包。

我们须要打开一个命令行窗体,并输入:

输入完这条命令后,你会得到一个安装界面例如以下:

下一步,下一步完毕安装后你会得到这种一个文件夹:

这个文件夹里有我们写JAXWS所需的全部lib包以及JAXWS自带的教程。

三、一切始于HelloWorld

3.1建立Webservice的Server端project

能够看到我们这个文件夹除传统的src,WebContent文件夹外还有几个文件夹,它们各自是:

?    build
?    wsdl
?    wssrc

我们来书写我们的第一个Webservice吧,它的名字叫Hello(Come on, 老套了,又来了)。

package ctsjavacoe.ws.fromjava;

import javax.jws.WebMethod;

import javax.jws.WebService;

@WebService

public class Hello {

@WebMethod

public String say(String name) {

return ("Hello: "+name);

}

}

注意:

@WebService

凝视在了Class之上,这告诉了JAXWS,此类为Webservice。

@WebMethod

凝视在了public方法上,这告诉了JAXWS,此方法为soap方法,该方法有两个參数,一个input的String,一个output的String。

业务逻辑非常easy,client调用传入一个Name,服务端返回给client一个”Hello: “+name的字串。

如今我们通过Java文件来生成Webservice相关布署文件以及调用接口。

3.2通过Java类编译Webservice

JAX-WS 2.0 有两种开发过程:自顶向下和自底向上。自顶向下方式指通过一个 WSDL 文件来创建Web Service,自底向上是从 Java 类出发创建 Web Service。两种开发过程终于形成的文件包含:

1.SEI。一个SEI相应WSDL中WebService的一个port,在Java中是一个Java接口。

2.SEI实现类。

3.WSDL和XSD文件。

结合公司内项目的特点,我们很多其它的是碰到以下两种情况:

1.  Onsite要我们做一个Webservice或者是客户要求我们提供Webservice接口;

2.  Onsite已经有一个Webservice了,如今要我们做client集成。

因此,我们选用Server端通过Java Class生成webservice,而client通过wsdl生成Java调用类的做法

JAXWS为我们提供了两个工具:

ü  wsgen

主要用于Server端通过Java类编译成Webservice及相关的wsdl文件

ü  wsimport

主要用于Client端(调用端)通过wsdl编译出调用Server端的Java文件

我们就来生成一下上面的这个Hello,打开一个command窗体,键入例如以下命令:

-wsdl參数代表生成webservice

- s參数代表生成的.java文件置于何处

-d 參数代表生成的编译class文件置于何处(这个能够忽略,我们利用eclipse编译)

-r 參数代表生成的.wsdl文件与.xsd文件生成在何处

-cp參数代表classpath,即Hello.class的所在,为什么我们的-cp后是这么长一个路径呢?请看Eclipse里project编译输出文件夹的路径就知道了:


3.2.1生成的src文件

好,我们如今回到eclipseproject里,刷新一下project:

看到在wssrc文件夹下已经生成了我们所需的java文件了,请手工cut(对,是cut)这些文件到我们project的”src”文件夹,假设不cut,下次继续使用该文件夹生成webservice类时,wsgen有时会生成不了,但也不报错,不知道为什么,查了一下,可能是一个bug,因该会在兴许的jdk1.6.30up里改进。

3.2.2生成的wsdl及xsd文件

这里我们有两个文件,一个是wsdl文件,这个就是我们的webservice的entry,一个是xsd文件,这个是什么?

这个就是我们java的方法里的參数的相应,或者换句话说,它就是xml格式的java bean,在webservice的世界里,xsd是作为数据结构描写叙述用的。

如今我们的webservice的服务端有了。

3.2.3布署webservice

布署前的准备:

在project的WEB-INF文件夹下建立“sun-jaxws.xml”文件,内容例如以下:

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

<endpoints xmlns='http://java.sun.com/xml/ns/jax-ws/ri/runtime'

    version='2.0'>

    <endpoint name='Hello' implementation='ctsjavacoe.ws.fromjava.Hello'

        url-pattern='/HelloService' />

</endpoints>

将ctsjavacoe.ws.fromjava.Hello声明为Web Service。

假设是从WSDL生成的Web Service,则写法为,

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

<endpoints version="2.0" xmlns="http://java.sun.com/xml/ns/jax-ws/ri/runtime">

<endpoint implementation="ctsjavacoe.ws.fromjava.HelloSEI"

name="Hello" url-pattern="/HelloService" />

</endpoints>

 

改动WEB-INF文件夹下的web.xml文件,增加例如以下内容:

<servlet>

        <servlet-name>Hello</servlet-name>

        <servlet-class>com.sun.xml.ws.transport.http.servlet.WSServlet</servlet-class>

        <load-on-startup>2</load-on-startup>

</servlet>

<servlet-mapping>

        <servlet-name>Hello</servlet-name>

        <url-pattern>/HelloService</url-pattern>

</servlet-mapping>

为每个WebService声明一个com.sun.xml.ws.transport.http.servlet.WSServlet。

開始布署:

1.  在tomcat的webapps文件夹下建立一个文件夹叫“D:\tomcat2\webapps\JaxWSSample”

2.  把eclipseprojectJaxWSSample下WebContent文件夹下全部的东西copy至该文件夹下

3.  重新启动tomcat

在ie中输入:

http://localhost:9090/JaxWSSample/HelloService?wsdl

能够看到我们的webservice已经生成了。


3.3通过Server端的WSDL生成供JAVA调用的client

3.3.1同步与异步

同步调用,非常好理解,即一来一回,Client端request到Server端,Sever端立马回一个response。

异步调用,就是client调用一次服务端后,服务端处理事务并非即时返回的,比方说传一个600MB文件给服务端,服务端在处理接收和解析文件时,client不会立即得到一个响应,它会等待一段时间,等server处理完后,再通知client“我处理完了”。

3.3.2利用wsimport产生client

我们新建一个eclipse的project,仅仅须要是JAVAproject即可了,不须要webproject的,由于我们这边仅仅用代码做调用:

把Server端的wsdl及xsd都手工copy到clientproject的wsdl文件夹下。

打开一个cmd窗体敲入例如以下的命令:

wsimport -keep -d bin -s src wsdl/HelloService.wsdl

以上是产生同步client的命令。

假设要产生异步client命令,须要在project根文件夹下建一个binding.xml的文件,内容例如以下:

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

<bindings xmlns:xsd="http://www.w3.org/2001/XMLSchema"

    xmlns:wsdl="http://schemas.xmlsoap.org/wsdl/" wsdlLocation="wsdl/HelloService.wsdl"

    xmlns="http://java.sun.com/xml/ns/jaxws">

    <bindings node="wsdl:definitions">

        <enableAsyncMapping>true</enableAsyncMapping>

    </bindings>

</bindings> 

然后产生client代码的wsimport命令也会不一样:

wsimport -keep –b binding.xml  -d bin -s src wsdl/HelloService.wsdl

我们来看异步调用的代码(同步代码比异步调用简单,留给大家自己做练习)

wsimport命令会在eclipseproject的src文件夹中生成你在调用时所用的java src文件。

Hello.java与HelloService.java是wsimport给我们生成的供client调用的java文件。

我们把HelloService文件打开,能够看到两行:

file:/D:/workspace/JaxWSClient/wsdl/HelloService.wsdl

把它们改成:

http://localhost:9090/JaxWSSample/HelloService?wsdl

有两行,尤其是Url url=这一行,千万不要漏改了。

我们创建一个调用类,叫:HelloAsyncPollingClient.java文件,内容例如以下:

package ctsjavacoe.ws.fromjava;

 

import javax.xml.ws.Response;

 

public class HelloAsyncPollingClient {

 

    /**

     * @param args

     */

    public static void main(String[] args) throws Exception {

        HelloService service = new HelloService();

        Hello port = service.getHelloPort();

        Response<SayResponse> sayAsync = port.sayAsync("Mk");

        while (!sayAsync.isDone()) {

            System.out.println("is not down");

        }

        try {

            SayResponse callNameResponse = sayAsync.get();

            String message = callNameResponse.getReturn();

            System.out.println(message);

        } catch (Exception ex) {

        }

    }

 

}

执行,得到结果例如以下:


3.3.3细说同步与异步

在旧的基于JAX-RPC的webservice编程model中,是不支持异步的service 调用的,在最新的Jax-ws webservice 编程model中,增加了对webservice的异步调用的支持。

首先我来讲一下它的原理,大家不要以为在异步的调用下,从client到server 之间的soap message 流也是异步的,事实上不是,Soap/Http 协议在同步跟异步的调用下是一样的,都是client的service在执行时打开一个connectin,发送请求,然后接收返回,这些都在同一个connection中。这种方式对我们有什么影响呢?从client程序的角度来讲,没有影响,client的编程模型是由WSDL中的messages跟port types 来定义的,仅仅要这些东西没有改变,request 跟response是不是在同一个Tcp/ip 的session 中来发送对与我们来说没由影响,然后从架构跟资源的角度来讲,对我们的影响就大了,把连接层的资源跟应用层的程序执行状态绑定起来是由很多弊端的,假如在异步调用时,程序执行出现了异常,将会导致连接层的资源被一直占用,这样会极大的影响我们程序的,稳定性,可靠性,资源的使用跟性能。

3.3.4异步的还有一种实现

上例中实现的是一种“polling方式的异步调用”,以下给出“callback”方式的异步调用client。

由于此callBack当请求发出去以后当前的这个connection就会关闭 ,为了达到測试的目的,增加了sleep,让client程序等待server端得返回。

callback类型的client要传入一个javax.xml.ws.AsyncHandler类型的匿名内部类,当soapMessage 到达时,Jax-ws会调handleResponse这种方法来处理response.

client測试代码例如以下:

package ctsjavacoe.ws.fromjava;

import javax.xml.ws.AsyncHandler;

import javax.xml.ws.Response;

public class HelloAsyncCallBackClient {

    public static void main(String[] args) throws Exception {

        HelloService service = new HelloService();

        Hello port = service.getHelloPort();

        port.sayAsync("Mk", new AsyncHandler<SayResponse>() {

            public void handleResponse(Response<SayResponse> res) {

                try {

                    SayResponse response = null;

                    response = res.get();

                    String message = response.getReturn();

                    System.out.println(message);

                } catch (Exception e) {

                    e.printStackTrace();

                }

            }

        });

        Thread.sleep(1000);

    }

}

5天学会jaxws-webservice编程第一天的更多相关文章

  1. Java基础:三步学会Java Socket编程

    Java基础:三步学会Java Socket编程 http://tech.163.com 2006-04-10 09:17:18 来源: java-cn 网友评论11 条 论坛        第一步 ...

  2. 学会JavaScript函数式编程(第1部分)

    摘要: JS函数式编程入门. 原文:学会使用函数式编程的程序员(第1部分) 作者:前端小智 Fundebug经授权转载,版权归原作者所有. 在这篇由多部分组成的文章中,接下来将介绍函数式编程的一些概念 ...

  3. 5天玩转C#并行和多线程编程 —— 第一天 认识Parallel

    5天玩转C#并行和多线程编程系列文章目录 5天玩转C#并行和多线程编程 —— 第一天 认识Parallel 5天玩转C#并行和多线程编程 —— 第二天 并行集合和PLinq 5天玩转C#并行和多线程编 ...

  4. [未完成]WebService学习第一天学习笔记

    [未完成]WebService学习第一天学习笔记[未完成]WebService学习第一天学习笔记

  5. Windows核心编程第一章.错误处理

    Windows核心编程第一章,错误处理. 一丶错误处理 1.核心编程学习总结 不管是做逆向,开始做开发.在Windows下.你都需要看一下核心编程这本书.这本书确实写得很好.所以自己在学习这本书的同时 ...

  6. Python 编程第一步

    Python  编程第一步 在前面的教程中我们已经学习了一些 Python3 的基本语法知识,下面我们尝试来写一个斐波纳契数列. # Fibonacci series: 斐波纳契数列 # 两个元素的总 ...

  7. [Real World Haskell翻译]第24章 并发和多核编程 第一部分并发编程

    第24章 并发和多核编程 第一部分并发编程 当我们写这本书的时候,CPU架构正在以比过去几十年间更快的速度变化. 并发和并行的定义 并发程序需要同时执行多个不相关任务.考虑游戏服务器的例子:它通常是由 ...

  8. How do I set the timeout for a JAX-WS webservice client?

    How do I set the timeout for a JAX-WS webservice client? up vote58down votefavorite 27 I've used JAX ...

  9. python012 Python3 编程第一步

    Python3 编程第一步在前面的教程中我们已经学习了一些 Python3 的基本语法知识,下面我们尝试来写一个斐波纳契数列.实例如下: #!/usr/bin/python3 # Fibonacci ...

随机推荐

  1. Altium Designer学习:PCB只看当前层

    1 Shift+S          这个快捷键能高亮当前层,并且使其他层变成灰色,见下图:       2 隐藏指定层 在图中右下角的地方,右键,会弹出一个选项条,选择Hide Layers,可以选 ...

  2. MySQL主从复制中断处理一例

    收到mysql主从中断报警邮件,马上登上服务器查看,发现是中继日志损坏. Show slave status\G,提示中继日志损坏,按以往的做法,根据提示重新指定合适的日志文件以及pos点.  Rel ...

  3. 【POJ】1692 Crossed Matchings

    经典DP,想了很久,开始想复杂了. #include <iostream> using namespace std; #define MAXNUM 100 int mymax(int a, ...

  4. Oracle系列之表空间

    涉及到表的处理请参看原表结构与数据  Oracle建表插数据等等 创建表空间等等 select tablespace_name from dba_tablespaces;--dba权限用户查询数据库中 ...

  5. URAL1009

    链接 第一道URAL题 简单递推 #include <iostream> #include<cstdio> #include<cstring> #include&l ...

  6. Nagios Core/Icinga 基于栈的缓冲区溢出漏洞

    漏洞名称: Nagios Core/Icinga 基于栈的缓冲区溢出漏洞 CNNVD编号: CNNVD-201402-484 发布时间: 2014-03-03 更新时间: 2014-03-03 危害等 ...

  7. $apply() $digest()

    理解Angular中的$apply()以及$digest() <!DOCTYPE html> <html> <head> <meta charset=&quo ...

  8. C# asp.net 操作Word的前提配置和简单的方法

    操作的前提: 1.要保证机器本身要安装OFFICE. 有时安装了Office,但是不能找到Microsoft Word 11.0(或者更高的版本) Object Library.那可能是因为在安装of ...

  9. 使用Spring MVC 的表单控制器SimpleFormController

    以注册过程为例,我们可能会选择继承AbstractController来实现表单的显示,继承AbstractCommandController来实现表单的处理 ,这样是可行的,但必须要维护两个控制器 ...

  10. LightOJ 1220 Mysterious Bacteria 水题

    暴力就行了,找出素因子,正的最多是30,然后负的最多是31(这一点wa了一次) #include <cstdio> #include <iostream> #include & ...