C++与Java通过WebService通信(下)

一、 前言

本篇讲述如何通过Java客户端访问C++服务端发布的SOAP模式的WebService接口。文档中的样例代码拷贝出去即可运行,所有的代码都是本地测试OK的;本文不但解决了接口调用的问题,同时解决了中文乱码的问题。

二、 环境准备

1、 CXF组件:Java端用于发布WebService服务的开源组件,内部自带jetty Web容器。
2、 Gsoap组件:C++端用户访问WebService服务的组件。使用的是gsoap-2.8。这两个组件可以直接google,官网下载。
3、 Eclipse:Java开发IDE。
4、 VS 2010:C++开发IDE。

三、 C++服务端开发

Step1 定义WebService接口文件helloWebService.h

//gsoap ns service name: wscplus
//gsoap ns service style: rpc
//gsoap ns service namespace: http://localhost:10010/wscplus.wsdl
//gsoap ns service location: http://localhost:10010
//gsoap ns schema namespace: urn:wscplus
int ns__hellowebservice(char* param, char** result);

定义接口需要注意:
1、 接口返回值是int型.
2、 接口名称定义格式为 ns__xxx。
3、 输入参数是字符串指针,输出参数是指针的指针。同样,可以通过json串传递更多的内容。
4、 函数头注释按给的样例定义。在注释中定义服务IP,端口,wsdl文件名。如果函数头未按指定格式定义,使用soapcpp2.exe转换时将产生不了wsdl文件。

Step2使用gsoap-2.8\gsoap\bin\win32\ soapcpp2.exe生成服务端代码。

技术分享

生成了如下文件:
技术分享

Step3 将头.h.cpp\nsmap拷贝到VS2010工程中,编译。

技术分享

编译不过,将soapServerLib.cpp中的#include “soapC.cpp”、#include “soapServer.cpp”注释掉。

Step4实现WebService接口

// 实现WebService接口

int ns__hellowebservice(struct soap* soapObject, char* param, char** result)
{
    wchar_t* param1 = MulityByteToWideChar(CP_UTF8, param);
    printf("接收到Java客户端传过来的参数-param: %s\n", WideCharToMulityByte(CP_ACP, param1));

    *result = WideCharToMulityByte(CP_UTF8, L"abKJLcd123e12一大堆中文输出参数");
    return SOAP_OK;
}

接口实现函数的第一个参数是struct soap* soapObject。后面的参数与helloWebService.h中定义的接口一致。为了保证中文传输不乱码,接收的参数和返回值都做了编码转换。

Step5 实现http_get函数,返回wsdl文件信息

// 编写get响应请求,目的是为了在浏览器中输入url能看到wsdl文件。
// 也方便在soapUI等Webservice接口调试工具中能直接调用定义的接口。

int http_get(struct soap* soapObject)
{    
    FILE*fd = NULL;

    // wscplus.wsdl 是执行soapcpp2.exe命令时生成的。把他拷贝到了当前目录下。
    fd = fopen("wscplus.wsdl", "rb"); //open WSDL file to copy        
    if (!fd)        
    {                
        return 404; //return HTTP not found error        
    }

    soapObject->http_content = "text/xml";  //HTTP header with text /xml content        
    soap_response(soapObject, SOAP_FILE);        
    for(;;)        
    {
        size_t r = fread(soapObject->tmpbuf,1, sizeof(soapObject->tmpbuf), fd);

        if (!r)                
        {
            break;                
        }                

        if (soap_send_raw(soapObject, soapObject->tmpbuf, r))
        {
            break; //cannot send, but little we can do about that                
        }
    }

    fclose(fd);
    soap_end_send(soapObject);       

    return SOAP_OK;
}

Step6 发布WebService服务

int _tmain(int argc, _TCHAR* argv[])
{
    struct soap soapObject;
    soap_init(&soapObject);
    soap_set_mode(&soapObject, SOAP_C_UTFSTRING);

    soapObject.fget = http_get;
    soap_set_namespaces(&soapObject, namespaces);

    int ret = soap_bind(&soapObject, NULL, 10010, 100);

    if (ret < 0)
    {
        return -1;
    }

    while (true)
    {
        // 阻塞线程,等待外部请求
        ret = soap_accept(&soapObject);

        if (ret <  0)
        {
            return -1;
        }

        soap_serve(&soapObject);
        soap_end(&soapObject);
    }

    return 0;
}

四、 Java客户端开发

Step1建好Java项目,导入CXF lib目录下的Jar包。

不能不说Java的开源组件太好用,傻瓜式开发。C++的gsoap太麻烦了。

Step2 开发客户端代码, 调用WebService接口。

import java.nio.charset.Charset;
import java.rmi.RemoteException;   

import javax.xml.namespace.QName;   
import javax.xml.rpc.ParameterMode;
import javax.xml.rpc.ServiceException;

import org.apache.axis.client.Call;   
import org.apache.axis.client.Service;
import org.apache.axis.encoding.XMLType;

public class StartupClient 
{
    public static void main(String[] args) 
    {
        try 
        {   
            String endpoint = "http://localhost:10010/wscplus?wsdl";    
            Service service = new Service();   

            Call call = (Call)service.createCall();    
            call.setTargetEndpointAddress(endpoint);  

            // 设置调用的接口,指定输入参数,输出参数
            call.setOperationName(new QName("urn:wscplus", "hellowebservice"));    
            call.addParameter("param", XMLType.XSD_STRING, ParameterMode.IN);
            call.addParameter("result", XMLType.XSD_STRING, ParameterMode.OUT);

            call.setReturnType(XMLType.XSD_STRING);
            String str = "237anastasiaABG293729一大堆中文输入参数";
            String strParam = new String(str.getBytes(Charset.forName("UTF-8")));
            String result = (String)call.invoke(new Object[]{strParam});
            System.out.println("获取C++服务端返回值-result: " + result);   
        }
        catch (ServiceException e) 
        {   
            e.printStackTrace();   
        } 
        catch (RemoteException e) 
        {   
            e.printStackTrace();   
        }
    }
}

五、 调试验证

先启动服务端,在eclipse中启动客户端单步调试,可以看输出参数,返回值信息。

郑重声明:本站内容如果来自互联网及其他传播媒体,其版权均属原媒体及文章作者所有。转载目的在于传递更多信息及用于网络分享,并不代表本站赞同其观点和对其真实性负责,也不构成任何其他建议。