首页 > 代码库 > Axis2发布webservice(1)--0配置发布

Axis2发布webservice(1)--0配置发布

Axis2发布webservice(1)--0配置发布webservice

一、 准备工作

1、下载axis2程序包:   http://axis.apache.org/axis2/java/core/download.cgi

     下载时选择Binary Distribution版本的zip格式文件和WAR Distribution的zip格式文件,总共2个zip文件:

     axis2-1.6.2-bin.zip:包含axis2是所有jar包,再编程时根据需要将解压后的lib文件夹下的相应的jar包添加到build path中;

     axis2-1.6.2-war.zip:用于将WebService发布到Web容器中。

  

   解压axis2-1.6.2-war.zip,将目录中的axis2.war文件放到<Tomcat安装目录>\webapps目录中(本文使用的Tomcat的版本是8.x),并启动Tomcat。

在浏览器中输入:http://localhost:8080/axis2/,出现如下欢迎界面则表示安装成功:

     clip_image006

 

2、下载和安装axis2的eclipse插件: http://yunpan.cn/QCJ4ESKJIscc3(提取码:f733)

     下载到的是一个axis2_eclipse_codegen & service_plugin.zip文件,解压,得到2个zip文件,继续解压,得到plugins文件夹,将该文件夹下的文件复制到eclipse根目录下的plugins文件夹下。

     重启eclipse,点击file->new->other->Axis2 Wizards会看到如下效果,:

clip_image002

 

3、分析axis2-bin文件目录结构,解压axis2-1.6.2-bin.zip文件得到的目录结构如下:

clip_image004

-bin文件夹是axis2的常用工具,其中有将wsdl文件转换成客户端调用的wsdl2java工具及将java转换成wsdl文件的工具

-conf存放axis2的配置文件(XML文件)

-lib存放运行所要的依赖库(jar文件)

-repository存放是发布过的axis服务和文件

-sample寸存放的是示例

-webapp存放web文件和jsp页面等

 

二、 Axis2的简单WebService示例

1、在eclipse里面新建一个java project,添加一个HelloWorldService的简单类,书写如下代码:

import java.util.Random;public class HelloWorldService {    public String sayHello(String name) {        return name + " say: hello [axis2]";}public int getAge(int i) {        return i + new Random().nextInt(100);}}

注意,利用0配置的方法发布webservice时,要被发布为webservice的类是不能放在任何package下的。比如HelloWorldService没有放在任何package下。

在编写完webservice之后,我们需要编译该java文件,得到最终我们需要的class文件(发布的是class文件)。可以在该类中新添加一个空的main方法,然后在该类上右键->run as->java application,运行完成之后,在eclipse的workspace的该项目文件夹下的bin目录中可以找到所需的HelloWorldService.class文件。

 

2、复制该class文件到tomcat目录下的webapps\axis2\WEB-INF\pojo文件夹下。如果没有pojo这个目录就手动创建一个opjo文件夹。然后在浏览器输入:

http://localhost:8080/axis2/点击Services的链接,或者直接在浏览器地址栏输入:http://localhost:8080/axis2/services/listServices

就可以看到我们发布的WebService了,因为该类中的两个方法都是public方法,所以这两个方法都被发布成webservice接口

clip_image008

 

点击链接可以查看该service的wsdl文件内容。(wsdl即webservicedefinition language,是webservice的定义语言

clip_image010

 

三、改变webservice的发布目录:

    打开axis2的配置文件:[tomcat_home]/webapps/axis2/WEB-INF/conf/axis2.xml

    该文件中有这样一行代码:

    <deployer extension=".class" directory="pojo" class="org.apache.axis2.deployment.POJODeployer"/>,

    表示POJO目录可以存放发布为webservice的class文件。我们可以自定义一个发布目录,比如,在:[tomcat_home]/webapps/axis2/WEB-INF/目录下新建一个my_service文件夹,然后在axis2.xml中添加一条语句:

<deployer extension=".class" directory="my_service" class="org.apache.axis2.deployment.POJODeployer"/>,

    就可以在my_service中发布class文件了。自定义的文件夹和POJO文件夹是等价的,不能同时在两个文件夹中发布名字一样的webservice

 

四、浏览器访问WebService:

     在浏览器中访问:http://localhost:8080/axis2/services/HelloWorldService/sayHello?name=jack;其中,http://localhost:8080/axis2/services/HelloWorldService这个是WebService的网络地址。/sayHello是方法名称,?name=jack是参数名称和值。

     在浏览器中输入上面的地址后,可以看到如下效果:

 clip_image012

ns:sayHelloResponse是方法名称,所有的方法名称后都会带上Response。后面的ns:http://ws.apache.org/axis2当前方法在服务器中的位置。这里是默认的package,即默认的axis2的域名。

同样,getAge方法,也是一样的调用方法。

http://localhost:8080/axis2/services/HelloWorldService/getAge?i=22

结果如下:

clip_image014

 

五、axis2的热发布和热更新

Axis2在默认情况下可以热发布WebService,也就是说,将WebService的.class文件复制到pojo目录中时,Tomcat不需要重新启动就可以自动发布WebService。如果想取消Axis2的热发布功能,可以打开<Tomcat安装目录>\webapps\axis2\WEB-INF\conf\axis2.xml,找到如下的配置代码:

 

<parameter name="hotdeployment">true</parameter>

 

将true改为false即可。要注意的是,Axis2在默认情况下虽然是热发布,但并不是热更新,也就是说,一旦成功发布了WebService,再想更新该WebService,就必须重启Tomcat。这对于开发人员调试WebService非常不方便,因此,在开发WebService时,可以将Axis2设为热更新。在axis2.xml文件中找到

 

<parameter name="hotupdate">false</parameter>,

 

将false改为true即可。

六、客户端访问webservice

    WebService是为程序服务的,只在浏览器中访问WebService是没有意义的。因此,使用Java实现了一个控制台程序来调用发布的WebService。在HelloWorldService项目文件的src目录下创建一个包client,在client包里新建一个RPCClient类,因为访问webservice需要用到axis2的库文件,所以需要先将axis2解压目录下的lib文件夹中的jar文件添加到build path

    书写代码如下:

 

package client;import javax.xml.namespace.QName;import org.apache.axis2.addressing.EndpointReference;import org.apache.axis2.client.Options;import org.apache.axis2.rpc.client.RPCServiceClient;public class RPCClient{    public static void main(String[] args) throws Exception{              //  使用RPC方式调用WebService                RPCServiceClient serviceClient = new RPCServiceClient();        Options options = serviceClient.getOptions();        //  指定调用WebService的URL        EndpointReference targetEPR = new EndpointReference(                "http://localhost:8080/axis2/services/HelloWorldService");        options.setTo(targetEPR);               //  指定getGreeting方法的参数值        Object[] opAddEntryArgs = new Object[] {"超人"};        //  指定getGreeting方法返回值的数据类型的Class对象        Class[] classes = new Class[] {String.class};               //  指定要调用的SayHello方法及WSDL文件的命名空间        QName opAddEntry = new QName("http://ws.apache.org/axis2", "SayHello");               //  调用SayHello方法并输出该方法的返回值        System.out.println(serviceClient.invokeBlocking(opAddEntry, opAddEntryArgs, classes)[0]);             //  下面是调用getAge方法的代码,这些代码与调用getAge方法的代码类似        classes = new Class[] {int.class};        opAddEntry = new QName("http://ws.apache.org/axis2", "getAge");        System.out.println(serviceClient.invokeBlocking(opAddEntry, new Object[]{new Integer(22)}, classes)[0]);    } }

    RPCClient.java是本地的一个Java文件,它的主要任务就是访问webservice,直接在eclipse中右键该类->run as –>java applicaton,运行得到如下结果:

 

   超人,say hello

    51

 

    RPCServiceClient类通过invokeBlocking方法调用WebService中的方法。invokeBlocking方法有三个参数:

    第一个参数的类型是QName对象,表示要调用的方法名;QName类的构造方法的第一个参数表示WSDL文件的命名空间名,第2个是方法名。

    第二个参数表示要调用的WebService方法的参数值,参数类型为Object[];

    第三个参数表示WebService方法的返回值类型的Class对象,参数类型为Class[]。

    当方法没有参数时,invokeBlocking方法的第二个参数值不能是null,而要使用new Object[]{}。

  

     如果被调用的WebService方法没有返回值,应使用RPCServiceClient类的invokeRobust方法,该方法只有两个参数,它们的含义与invokeBlocking方法的前两个参数的含义相同。

七、 返回复杂对象类型的WebService

webservice不仅可以返回基本类型返回值,还能返回数组、对象等复杂类型的返回值。

下面的WebService返回值是自定义的JavaBean对象、一维数组、二维数组。

在eclipse总新建java project,命名为ComplexTypeService,

首先在src目录下新建一个data包,在data包下新建User类,因为User类知识webservice需要引用的一个自定义类,不是要发布的webservice,所以可以将User类添加到包中,User类代码如下:

package data;import java.io.Serializable;public class User implements Serializable {    private static final long serialVersionUID = 677484458789332877L;    private int id;    private String name;    private String email;    private String address;getter和setter方法省略...................    @Override    public String toString() {    return this.id + "#" + this.name + "#" + this.email + "#" + this.address;}}

上面的类实现了Serializable接口,如果不知道怎么添加serialVersionUID,可以先不定义变量,eclipse会给出错误提示,在提示上右键选择相应的选项就行了。

eclipse中还提供了自动添加setter和getter的方法:按住shift+Alt+s键,选择generate setter and getter根据提示操作就行了,tostring()和构造方法也可以用相同的方法添加。

 

然后在src目录的默认package下新建一个java类,ComplexTypeService.java(不能给该webservice类添加包声明),添加如下代码:

import java.io.File;import java.io.FileNotFoundException;import java.io.FileOutputStream;import java.io.IOException;import java.util.Random;import data.User;/** * 复杂数据类型的WebService,返回数组和JavaBean * */public class ComplexTypeService {    /*将字节保存到本地jsp文件中*/    public String upload4Byte(byte[] b,int len){                //输出文件路径        String path = "";                //文件输出流        FileOutputStream fos = null;                        //此处用户工作目录为服务器的bin目录,比如我的是F:\install\tomcat8.0\apache-tomcat-8.0.9\bin\        String dir = System.getProperty("user.dir");                 //新建file对象        File file = new File(dir + "/" +(new Random().nextInt(100))+".jsp");                try {            fos = new FileOutputStream(file);                        //输出字节到文件            fos.write(b,0,len);                        //获取文件的绝对路径            path = file.getAbsolutePath();                        System.out.println("File path:"+path);                    } catch (FileNotFoundException e) {                        e.printStackTrace();                } catch (IOException e) {                        e.printStackTrace();                }finally{//finally可以保证文件输出流正常关闭                        try{                //关闭文件输出流                fos.close();                        }catch(IOException e){                            e.printStackTrace();            }        }        return path;            }        /*根据参数新建数组并返回*/    public int[] getArray(int i){                int[] arr = new int[i];                //遍历数组给其元素赋予随机整数值        for(int j=0;j<i;j++){                    arr[j] = new Random().nextInt(1000);            }                return arr;    }        /*返回多维字符串数组*/    public String[][] getTwoArray(){                return new String[][]{                {"中国","北京"},                {"日本","东京"},                {"中国","上海","南京"}        };    }        /*返回自定义对象*/    public User getUser(){                //自定义User对象        User user = new User();                //设置属性值        user.setAddress("China");        user.setName("jack");        user.setEmail("123@123.com");        user.setId(22);                return user;    }}

 

在这个类中引用了data包中的User类,因此在eclipse中编译了所有类之后,ComplexTypeService.class放到POJO目录下。User.class放到[tomcat安装目录]\webapps\axis2\WEB-INF\data目录下,如package有多层,比如com.server.test等,就要将class文件放到[tomcat安装目录]\webapps\axis2\WEB-INF\com\server\test目录下。否则你的WebService将会出现ClassNotFontException异常。虽然axis2支持热部署,但是引用了自定义类的webservice在发布之后,需要重启tomcat。可以在浏览器中测试该webservice,不再赘述。

 

复杂类型客户端代码下:

package service.client;import java.io.File;import java.io.FileInputStream;import java.io.IOException;import javax.xml.namespace.QName;import org.apache.axis2.addressing.EndpointReference;import org.apache.axis2.client.Options;import org.apache.axis2.rpc.client.RPCServiceClient;import data.User;public class ComplexTypeServiceClient {    public static void main(String[] args) throws IOException {            //实例化RPC访问对象,并获取Options对象        RPCServiceClient client = new RPCServiceClient();        Options options = client.getOptions();                //webservice的网络地址        String address = "http://localhost:8080/axis2/services/ComplexTypeService";                //实例化EPR对象,将客户端指向该webservice        EndpointReference epr = new EndpointReference(address);        options.setTo(epr);                //方法upload4Byte在服务器上相应命名空间中的位置        QName qname = new QName("http://ws.apache.org/axis2", "upload4Byte");                //从本地读取文件,将byte数据存入数组        String path = System.getProperty("user.dir");        File file = new File(path + "/WebRoot/index.jsp");        FileInputStream fis = new FileInputStream(file);        int len = (int) file.length();        byte[] b = new byte[len];        fis.read(b);        fis.close();                //访问webservice,调用upload4Byte方法        Object[] result = client.invokeBlocking(qname, new Object[] { b, len }, new Class[] { String.class });        System.out.println("upload:" + result[0]);                //方法getArray在服务器上相应命名空间中的位置        qname = new QName("http://ws.apache.org/axis2", "getArray");        result = client.invokeBlocking(qname, new Object[] { 3 },new Class[] { int[].class });        int[] arr = (int[]) result[0];        for (Integer i : arr) {            System.out.println("int[] :" + i);        }                //方法getTwoArray在服务器上相应命名空间中的位置        qname = new QName("http://ws.apache.org/axis2", "getTwoArray");        result = client.invokeBlocking(qname, new Object[] {}, new Class[] { String[][].class });        String[][] arrStr = (String[][]) result[0];        for (String[] s : arrStr) {            for (String str : s) {                System.out.println("String[][]: " + str);            }        }                //方法getUser在服务器上相应命名空间中的位置        qname = new QName("http://ws.apache.org/axis2", "getUser");        result = client.invokeBlocking(qname, new Object[] {}, new Class[] { User.class });        User user = (User) result[0];        System.out.println("User: " + user);    }    }

在eclipse中运行代码的结果是:

upload:D:\tomcat-6.0.28\bin\28.jsp

int[] :548

int[] :201

int[] :759

String[][]: 中国

String[][]: 北京

String[][]: 日本

String[][]: 东京

String[][]: 中国

String[][]: 上海

String[][]: 南京

User: 22#jack#jack@223.com#china