c++调用webservice接口的几种方法以及实例
前言
最近在做这方面的工作,过程中遇到一些问题,故在此整理并分享一下,让后来者少走一些弯路.废话不多说,直接上干货.
c++调用webservice的几种方法
目前,常用的调用方法有三种(个人知识有限,欢迎提点),本文中主要介绍第一和第三种方法:
- 使用sproxy.exe生成头文件,此种方法最简单,直接加载到项目中调用即可;
- 使用wsdl.exe生成头文件,加载到项目中调用,与第一点同理,但未尝试,有空再回来补充;
- 使用gsoap生成文件,把生成的相关文件均加载到项目中调用(此方法较第一点比较麻烦);
关于webservice的准备工作
首先你得有webservice地址,然后你才能够调用.
网上有很多公共的webservice接口,这里用其中一个"国内手机号码归属地查询WEB服务"来举个栗子.
一般的接口是这样的: http://www.webxml.com.cn/WebServices/MobileCodeWS.asmx
第一步,添加后缀
添加后缀 ?wsdl 得到如下网址:
http://www.webxml.com.cn/WebServices/MobileCodeWS.asmx?wsdl 打开是这样,说明成功
第二步,保存文件
把网页上的这些文件另存为wsdl格式的文件,右键->另存为即可下载.
下载完成.
方法一:使用sproxy.exe生成头文件并使用
1.生成头文件
sproxy使用方法: sproxy.exe /wsdl /out:“生成头文件目录名称” “生成的wsdl文件的路径”
eg:我的文件放在F:\webservice.WSDL转换成C转换成C++代码中,使用cmd进入此文件夹,并输入命令:
F:\webservice.WSDL转换成C转换成C++代码> .\sproxy.exe /wsdl /out:"F:\webservice.WSDL转换成C转换成C++代码\MobileCodeWS.h" "F:\webservice.WSDL转换成C转换成C++代码\MobileCodeWS.wsdl"
执行完成后,会生成MobileCodeWS.h文件.
2.使用接口实例
观察一下生成的.h中要使用的接口:
有三个参数,最后一个参数是返回的结果.
1.首先添加头文件以及使用命名空间
#include "MobileCodeWS.h"
using namespace MobileCodeWS;
2.实例化
CMobileCodeWS *oMobileWS = new CMobileCodeWS;
3调用以及传参
int iRet = oMobileWS->getMobileCodeInfo((CComBSTR)"15481348245",(CComBSTR)"",&bsRet);
4转换
//BSTR转换方法使用_com_util::ConvertBSTRToString(bsRet)
注意:
- CComBSTR头文件:<comdef.h>
- 使用_com_util::ConvertBSTRToString需添加#pragma comment(lib, “comsuppw.lib”)
方法二:使用wsdl.exe
这个方法可能适合c#使用,c++未尝试,待更.
使用方法类似方法一,自行研究.
方法三:使用gsoap.exe生成文件
由于未知原因,在使用第一种方法时,第一步生成头文件就失败,无法解析wadl的内容,报error.因此本人在网上搜罗大牛的博客查询资料借鉴整理验证,集前人所得才可最终成功.此方法稍微复杂一点点点点,各位看官请耐心.
1.使用wsdl2生成头文件
注意:
如果说传输的内容中含有中文,则需要增加一个bat文件,把传输类型由std::string转为wchar_t*,由宽字符传输,则可以解决乱码问题.刚开始由于没有添加处理文件导致后续接收到的中文乱码,折腾了半天,希望大家少走弯路.
bat文件(用txt文本保存如下文件,再修改后缀为bat即可)中一句话:
xsd__string = | std::wstring | wchar_t*
在gsoap-2.8\gsoap\bin\win32目录下,首先使用wsdl2.exe生成头文件:
F:\gsoap-2.8\gsoap\bin\win32> .\wsdl2h.exe -o .\MobileCodeWS.h .\MobileCodeWS.wsdl
或者不生成wsdl文件,直接使用wsdl的网址也可以生成
F:\gsoap-2.8\gsoap\bin\win32> .\wsdl2h.exe -o .\MobileCodeWS.h http://www.webxml.com.cn/WebServices/MobileCodeWS.asmx?wsdl
添加bat项:
F:\gsoap-2.8\gsoap\bin\win32> .\wsdl2h.exe -o .\MobileCodeWS.h .\MobileCodeWS.wsdl -t chinano1.bat
2.使用soapcpp2生成系列文件
PS F:\gsoap-2.8\gsoap\bin\win32> .\soapcpp2.exe -C .\MobileCodeWS.h -I F:\gsoap-2.8\gsoap\import -x
soapcpp2常用选项
-C 仅生成客户端代码
-S 仅生成服务器端代码
-L 不要产生soapClientLib.c和soapServerLib.c文件
-c 产生纯C代码,否则是C++代码(与头文件有关)
-I 指定import路径
-x 不要产生XML示例文件
-i 生成C++包装,客户端为xxxxProxy.h(.cpp),服务器端为xxxxService.h(.cpp)。
执行成功后会生成红框中的文件
3.使用实例
1.加载入项目中
把上图红框中的五个文件(除了soapClientLib.cpp),以及gsoap下的stdsoap2.h和stdsoap2.cpp两个文件添加到项目中.
在项目中添加这七个文件后,cpp文件需要修改属性:在项目中的cpp上右键属性,选择不使用预编译头.
2.调用流程
首先需要添加头文件:
#include “soapH.h”
#include “MobileCodeWSSoap.nsmap”
实例如下:
#include "stdafx.h"
#include <iostream>
#include "soapH.h"
#include "MobileCodeWSSoap.nsmap"
#include <fstream>
using namespace std;
int main ()
{
// SOAP的客户端
struct soap m_oSoap;
// WebService调用对象
class _ns1__getMobileCodeInfo getMobileCodeObject;
// WebService返回对象
class _ns1__getMobileCodeInfoResponse getMobileCodeResponseObject;
// SOAP初始化
soap_init(&m_oSoap);
// 调用函数的参数赋值
char strName[24] = {0};
sprintf(strName,"15742451354");
wchar_t *wstrName = A2W(strName);
getMobileCodeObject.mobileCode = wstrName;
/*如果生成头文件时没有修改类型,那么用如下方式赋值,接收的类型当然也是string:
std::string strName = "15742451354";
getMobileCodeObject.mobileCode = strName;
*/
// 发送WebService请求,并获得返回结果
int nResult = soap_call___ns1__getMobileCodeInfo(&m_oSoap,NULL,NULL,&getMobileCodeObject,getMobileCodeResponseObject);
// 操作成功
if(SOAP_OK == nResult)
{
// 输出返回结果
wchar_t* strResult = getMobileCodeResponseObject.getMobileCodeInfoResult;
wcout.imbue(locale("chs"));
wcout<<"wcout:"<<strResult<<endl;
}
/// 关闭SOAP
soap_destroy(&m_oSoap);
soap_end(&m_oSoap);
soap_done(&m_oSoap);
system("pause");
}