JAVA 快速实现webService (同步逻辑)

今天因为需要调用第三方的接口些fake service, 用于testcase. 该第三方使用的是soap的webservice模式。

首先是创建service。

package myws;

import java.io.IOException;

import javax.jws.WebMethod;
import javax.jws.WebParam;
import javax.jws.WebResult;
import javax.jws.WebService;
import javax.xml.bind.annotation.XmlAccessType;
import javax.xml.bind.annotation.XmlAccessorType;
import javax.xml.bind.annotation.XmlRootElement;
import javax.xml.ws.Endpoint;

/**
 *
 * @author 帐前卒
 *
 */
@WebService(
        name="HELLO",
        targetNamespace="http://chillyc.info/api", 
        serviceName="API", 
        portName="PortName")
public class WebServiceHolder {
    @WebMethod
    @WebResult(name="return")
    public String hello(@WebParam(name="name")String name) {
        return "hello" + name;
    }
    
    public static void main(String[] args) throws IOException {
        Endpoint.publish("http://localhost:80/fake/ws", new WebServiceHolder());
        System.in.read();
    }
}

这里写System.in.read();是希望服务在那里卡死。基本上所有的server都是类似死循环的写法。所以我这里就偷懒使用IO.

这里要注意的是@WebService annotation. 其中 name是指这个portType 叫什么。
targetNameSpace这个在所有的后续调用中名字都是一致的。serviceName就是服务的名称。portName
其实就是提供服务的端口名称(这里对服务本身的调用没有什么关系)。如果没有name, 那么java会默认使用WebServiceHolder
也就类名称作为name.

运行后,在浏览器中打开

http://localhost:80/fake/ws?wsdl

然后就可以看到wsdl文件。

This XML file does not appear to have any style information associated with it. The document tree is shown below.
<!--
 Published by JAX-WS RI at http://jax-ws.dev.java.net. RI's version is JAX-WS RI 2.1.6 in JDK 6. 
-->
<!--
 Generated by JAX-WS RI at http://jax-ws.dev.java.net. RI's version is JAX-WS RI 2.1.6 in JDK 6. 
-->
<definitions xmlns:soap="http://schemas.xmlsoap.org/wsdl/soap/" xmlns:tns="http://chillyc.info/api" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns="http://schemas.xmlsoap.org/wsdl/" targetNamespace="http://chillyc.info/api" name="API">
<types>
<xsd:schema>
<xsd:import namespace="http://chillyc.info/api" schemaLocation="http://localhost/fake/ws?xsd=1"/>
</xsd:schema>
</types>
<message name="hello">
<part name="parameters" element="tns:hello"/>
</message>
<message name="helloResponse">
<part name="parameters" element="tns:helloResponse"/>
</message>
<message name="getReturnInfo">
<part name="parameters" element="tns:getReturnInfo"/>
</message>
<message name="getReturnInfoResponse">
<part name="parameters" element="tns:getReturnInfoResponse"/>
</message>
<portType name="HELLO">
<operation name="hello">
<input message="tns:hello"/>
<output message="tns:helloResponse"/>
</operation>
<operation name="getReturnInfo">
<input message="tns:getReturnInfo"/>
<output message="tns:getReturnInfoResponse"/>
</operation>
</portType>
<binding name="PortNameBinding" type="tns:HELLO">
<soap:binding transport="http://schemas.xmlsoap.org/soap/http" style="document"/>
<operation name="hello">
<soap:operation soapAction=""/>
<input>
<soap:body use="literal"/>
</input>
<output>
<soap:body use="literal"/>
</output>
</operation>
<operation name="getReturnInfo">
<soap:operation soapAction=""/>
<input>
<soap:body use="literal"/>
</input>
<output>
<soap:body use="literal"/>
</output>
</operation>
</binding>
<service name="API">
<port name="PortName" binding="tns:PortNameBinding">
<soap:address location="http://localhost/fake/ws"/>
</port>
</service>
</definitions>

大家自己对照刚才的那些name, serviceName等,在wsdl文件中的什么地方。

然后是写一个stub作为调用的接口。

package myws;

import javax.jws.WebParam;
import javax.jws.WebService;

/**
 *
 * @author 帐前卒
 *
 */
@WebService(targetNamespace = "http://chillyc.info/api", name="HELLO")
public interface WebServiceAPI {
    String hello(@WebParam(name="name")String name);

}

这里要注意的是 那个hello函数,必须与webService发布的函数名相一致(要看wdsl文件中的名字。)
另外WebParam中的name也需要和发布函数中的参数名字一致。
这里WebService中传入了两个值。其中name就是刚才WebService中的name. 其实就是wsdl中的portType.
如果这里写错了。就会有Undefined port type:{http://chillyc.info/api}Name. 这个错误。所以要小心。

另外写个可执行的类。

package myws;

import java.net.MalformedURLException;
import java.net.URL;

import javax.xml.namespace.QName;
import javax.xml.ws.Service;

/**
 *
 * @author 帐前卒
 *
 */
public class Client {
    public static void main(String[] args) throws MalformedURLException {
        WebServiceAPI api = Service.create(
                new URL("http://localhost:80/fake/ws?wsdl"),
                new QName("http://chillyc.info/api", "API"))
                .getPort(WebServiceAPI.class);
        System.out.println(api.hello("sss"));
    }
}

这里URL中的就是wsdl文件的地址。 QName传入的就是targetNamespace 和 serviceName.
另外getPort就填入刚才的stub. 然后直接调用stub中的接口就能得到结果。

done.

简单快速,搭建和写client 也就是10分钟搞定。当然这只是起步

这个方法适用于JDK6及以上,其他版本未知。