http://www.ibm.com/developerworks/cn/webservices/ws-xml-rpc/

引言

Internet 现在的受欢迎程度越来越高,由于这个原因及其固有的优势,促使开发人员和 IT 部门开始着手将复杂的 C/C++ 业务和科学应用程序向基于 Web 的环境迁移。简单对象访问协议(Simple Object Access Protocol,SOAP)、代表性状态传输(Representational State Transfer,REST)以及 XML 远程过程调用协议(XML Remote Procedure Call,XML-RPC)等 Web 服务协议可帮助将此类遗留应用程序集成到万维网,例如,可以使用 XML-RPC 作为将现有 C/C++ 程序与其他客户端技术集成的机制。这篇文章将帮助您确定何时选择 XML-RPC 而不使用 SOAP 与 REST。另外,本文还提供了详细步骤指南以及使用开放源代码 XML-RPC 库的 C++ 集成示例代码片段。

为什么选择 XML-RPC?

集成 C/C++ 的挑战可以通过多种方式加以解决。C/C++ 代码集成的典型方法包括通过公共对象请求代理体系结构(Common Object Request Broker Architecture,CORBA)、分布式组件对象模型(Distributed Component Object Model,DCOM)、远程方法调用(Remote Method Invocation,RMI)Internet ORB互联协议(Internet Inter-ORB Protocol,IIOP)以及 Java™ 本机接口(Java™ Native Interface,JNI)等进行集成。

图 1 显示了利用现有 C++ 代码集成(使用上面提到的典型方法)的采用不同编程语言(Java、VC++、PL/1)开发的三个不同应用程序。

图 1. 没有 XML-RPC 的当前方案

正如您看到的,C++ 代码应为每个 RMI/IIOP/JNI、CORBA 和 DCOM 客户端集成技术公开相应的接口。这就要求进行三次开发工作,显然会使部署和管理此类复杂接口的过程变得更为困难和麻烦。

在这些情况下,XML-RPC 是一个更好的选择,因为它可能帮助简化开发、部署和管理工作。

关于 XML-RPC 和 REST

XML-RPC 是 UserLand Software 的 Dave Winer 于 1998 年提出的。请访问 Dave Winer 的博客,以了解他关于 XML-RPC 的最初想法。

REST 最初是由 Roy Thomas Fielding 作为其在加州大学尔湾分校的博士论文的一部分提出的。

图 2. 使用 XML-RPC

图 2 显示了如何使用 XML over HTTP 通过远程过程调用来调用 C++ 程序。SOAP 和 REST 等备选技术也能用于相同的目的。不过,您将在下一部分中了解到,这些技术之间存在一些关键区别。

SOAP、XML-RPC 及 REST 间的对比

尽管这三个协议都支持 XML-RPC over HTTP,但就 C++ 而言,它们彼此之间是有区别的。表 1 提供了这些协议的各个元素间详细的比较。

表 1. SOAP、XML-RPC 及 REST 间的对比
  SOAP XML-RPC REST
定义 SOAP 是用于在分散的分布式环境中进行信息交换的轻量级协议。该协议基于 XML,包括三个部分:信封、一组编码规则以及用于表示远程过程调用和响应的约定。 这是使用 HTTP 作为传输协议和使用 XML 作为编码方式的远程过程调用。XML-RPC 的设计力求简单,并同时允许传输、处理和返回复杂数据结构。 代表性状态传输旨在反映设计良好的 Web 应用程序的行为图像:由网页组成的网络,其中用户通过选择链接继续进行使用,选择链接将导致将下一页传输给用户并进行呈现,以供其使用。
目标 SOAP 可实现用户定义的数据类型,提供指定接收者的功能、消息特定的处理控制以及其他功能,从而对 XML-RPC 进行了扩展。 非常简单且条理清楚的可扩展格式。HTML 编码人员应该能够查看包含 XML-RPC 过程调用的文件,理解其进行的工作,并能够对其进行修改,只需一两次尝试就能使其正常工作。此协议非常容易实现,可以快速对其进行调整,以在其他环境或其他操作系统上运行。 创建 REST 的目的是为了提供有关 Web 应该如何运行的设计模式,并作为 Web 标准和设计 Web 服务的指导框架。
支持的数据类型 整数、Boolean、ASCII 字符串、双精度带符号浮点数、日期时间、结构、数组、字节数组、枚举、用户定义数据类型、多态访问器 整数、Boolean、ASCII 字符串、双精度带符号浮点数、日期时间、结构、数组。 特定于实现。通常支持的类型有整数、Boolean、ASCII 字符串、双精度带符号浮点数、日期时间、集合、列表、属性。
简单性 比 XML-RPC 略微复杂 易于理解和进行开发 特定于实现
稳定性 W3C 接受的标准 不是标准 体系结构参考。不需要标准
互操作性 无法与 REST/XML-RPC 互操作 无法与 REST/SOAP 互操作 无法与 SOAP/XML-RPC 互操作
工具 很多主要公司(包括 IBM 和 Microsoft)均已开始在其工具中支持 SOAP。 工具仍在开发中。 并没有很多工具支持。
自定义能力 高度可自定义,不受数据类型和协议限制 轻量级,只能在 HTTP 上工作,具有有限的数据类型支持 只能在 HTTP 上工作
有很多开放源代码库可用。 有很多开放源代码库可用。将在下面的部分进行说明 并没有很多实现库可用
表 2. C++ 的各种 XML-RPC 实现
库和包名称 说明
PDEL Packet Design Embedded Library 是一个 C 库,其中通过 http_xml_send_xmlrpc 和 http_servlet_xmlrpc 方法包含了客户机和服务器的 XML-RPC 实现。这些函数将帮助通过 HTTP 传输协议发送自定义 XML 数据。这个包还包含很多 XML-RPC 之外的其他功能。
XMLRPC++ 这是 XML-RPC 的 C++ 实现。其中提供了简单的服务器和客户机。通过使用面向对象的技术,我们可以集成这些服务器和客户机类,并实现我们自己的 XML-RPC 服务器,以将业务功能作为服务公开。在本文中,我们的示例实现及相关示例将使用这个库。
XMLRPC-C 这是一个 C 实现,可供 C 和 C++ 应用程序用于将这些方法作为服务公开。此包中包括一个 abyss Web 服务器。为了公开 C++ 方法,我们可以为所需的 C++ 方法编写 C 样式的包装,然后使用此库公开此方法。

安装 XML-RPC++ 库

为 C++ 程序启用 XML-RPC 的第一步是下载并安装 XML-RPC 库实现。为 C++ 程序提供了各种实现。可以在参考资料部分获得一个指向更多实现的链接。

我们的示例程序将使用 XMPLRPC++ 实现。有关将此实现下载并安装到 Linux、AIX、32 位 Windows 平台以及其他类似平台的详细信息,请参阅参考资料部分。我们的示例实现将基于 Red Hat 9 平台,使用的是 XML-RPC++ 0.7 库。

示例 C++ 应用程序

此处的示例应用程序是一个简单的两个整数相加的操作,将使用名为“operations”的用户定义类。清单 1 显示了 operations 类的代码片段。

您知道吗?

可以使用 Eclipse 作为开发 C/CPP 应用程序的 IDE。有关更多信息,请访问 Eclipse CDT

清单 1. Operations
  1. class operations {
  2. public:
  3. int add();
  4. operations(int i, int j);
  5.  
  6. private:
  7. int op1;//Operand 1
  8. int op2;//Operand 2
  9. };

该类的构造函数接受两个整数参数,并将其分别设置为私有变量 op1 和 op2。该类的 add 方法如清单 2 中所示。这个方法就是要作为 XML-RPC 服务公开的方法。

清单 2. Operations.cpp
  1. int operations::add()
  2. {
  3. std::cout << "Sum of "<<op1<<" + "<<op2<<" = "<<op1+op2<<std::endl;
  4. return(op1 + op2);
  5. }
 

回页首

XML-RPC 库的组件

在此部分,我们将使用一个类关系图来说明 XML-RPC 库的各个组件,并介绍其如何与我们服务器端的 operations 类通过接口连接。

图 3. XML-RPC 库和示例应用程序的类关系图

表 3 对每个类进行了详细说明。

表 3. 类详细信息
类名称 用途
Operations 要公开的 add 方法在该类中实现
Add 调用 operations 的 add 方法的包装类。该类也从 myXmlRpcServerMethod 继承
myXmlRpcServerMethod 该类从 XML-RPC 库的 XmlRpcServerMethod 类继承。该类的 execute 将在 Add 类中通过继承覆盖。
xmlRpcServermethod 需要向服务器注册的每个方法都必须通过 myXmlRpcServerMethod 类从该类进行继承,并实现自己的 execute 方法。这个 execute 方法将为公开的实际服务的包装。服务器收到 XML-RPC 调用时,将会直接触发此包装类的 execute 方法。在我们的示例中,Add 将为包装类,从客户端调用“Add”服务时,将调用其 execute 方法。
myXmlRpcServer 该类具有两个重要的私有变量
  1. pm_serverMethods:指向在服务器中注册的 myXmlRpcServerMethods 的指针列表。
  2. pm_xmlRpcServer:用于设置服务器 IP、端口和其他属性。

三个重要的方法

  1. Class constructor:使用 IP/端口详细信息初始化服务器对象,并将其绑定。
  2. pm_registerMethods:创建指向 Add 类的指针对象,并将其加入列表 pm_serverMethods 中。
  3. run:xmlRpcServer 类中 work 方法的包装
xmlRpcServer 该类是创建服务器对象的 XML-RPC 服务器类。该类具有以下两个重要的方法
  1. bindAndListen(port):绑定并侦听指定的特定端口
  2. work(...):启动服务器

清单 3 显示了上表中每个类的每个 .cpp 文件的代码。由于 xmlRpcServer 和 xmlRpcServerMethod 是在 XML-RPC 库中实现的,因此我们将重点讨论剩下的四个类。

清单 3. myXmlRpcServer.cpp
  1. #include "myXmlRpcServer.h"
  2.  
  3. using namespace XmlRpc;
  4. using namespace std;
  5.  
  6. myXmlRpcServer::myXmlRpcServer()
  7. {
  8. //call register methods
  9. pm_registerMethods();
  10.  
  11. //set port bind and listen
  12. int port = 8085;
  13. pm_xmlRpcServer.bindAndListen(port);
  14. std::cout<<"XmlRpcSever running in port "<<port<<std::endl;
  15. }
  16.  
  17. void
  18. myXmlRpcServer::pm_registerMethods()
  19. {
  20. Add* a=new Add(&pm_xmlRpcServer);
  21. myXmlRpcServerMethod *p=a;
  22. pm_serverMethods.push_back(p);
  23. }
  24.  
  25. void
  26. myXmlRpcServer::run()
  27. {
  28. pm_xmlRpcServer.work(-1);
  29. }
清单 4. myXmlRpcServer.h
  1. #include <iostream>
  2. #include "myXmlRpcServerMethods.h"
  3. #include "XmlRpc.h"
  4.  
  5. class myXmlRpcServer {
  6. public:
  7. myXmlRpcServer();
  8. void run();
  9. private:
  10. void pm_registerMethods();
  11. XmlRpc::XmlRpcServer pm_xmlRpcServer;
  12. std::list< myXmlRpcServerMethod* > pm_serverMethods;
  13. };

清单 5 和清单 6 显示了用于将方法注册到 XmlRpc Server(作为 XML-RPC 库的一部分提供)的类的代码。

清单 5. myXmlRpcServerMethods.cpp
  1. #include <iostream>
  2. #include "myXmlRpcServer.h"
  3. #include "operations.h"
  4.  
  5. using namespace std;
  6.  
  7. Add::Add(XmlRpcServer* s) : myXmlRpcServerMethod("Add", s) {};
  8.  
  9. Void Add::execute(XmlRpcValue & params, XmlRpcValue& result)
  10. {
  11. operations a(10,12);
  12. try
  13. {
  14. cout << "Inside Add::execute method\n";
  15. result = a.add();
  16. }
  17. catch(std::exception & stde)
  18. {
  19. throw XmlRpcException(stde.what());
  20. }
  21. }
清单 6. myXmlRpcServerMethods.h
  1. class myXmlRpcServerMethod : public XmlRpcServerMethod
  2. {
  3. public:
  4. myXmlRpcServerMethod
  5. (const char *name, XmlRpcServer * server):XmlRpcServerMethod(name, server) {}
  6. virtual void execute(XmlRpcValue & params, XmlRpcValue& result) {assert(0);}
  7. };
  8.  
  9. class Add:public myXmlRpcServerMethod
  10. {
  11. public:
  12. Add(XmlRpcServer* s);
  13. virtual void execute(XmlRpcValue & params, XmlRpcValue& result);
  14. };

服务器驱动程序

服务器端的入口点将是服务器驱动程序。将从此处实例化 myXmlRpcServer 对象,并调用 run() 方法,而后者将最终启动服务器。

清单 7. myServerDriver.cpp
  1. #include <iostream>
  2. #include "myXmlRpcServer.h"
  3.  
  4. int main(int argc, char* argv[])
  5. {
  6. myXmlRpcServer GeeBoomBaa;
  7. std::cout<<"About to run the server\n";
  8. GeeBoomBaa.run();
  9. return 0;
  10. }
 

回页首

启动服务器

为了编译该代码,请记住包含 (XML_RPC_INSTALL_DIR)/src 和 (XML_RPC_INSTALL_DIR)/include 目录。为了进行链接,请包含 libXmlRpc.a 库。代码成功编译并链接后,将获得一个可执行文件,该文件就是 XML-RPC 服务器。在我们的示例实现中,服务器将运行于 localhost 上,并侦听端口 8085。这个设置硬编码在 myXmlRpcServer.cpp 文件中。也可以使程序读取配置文件,或者从命令行提示符将此作为参数传递给程序。运行成功进行了编译和链接后得到的 a.out 程序,以启动服务器。

 

回页首

示例客户机

示例客户机应该从 XML-RPC 库提供的 XmlRpcClient 类实例化一个对象。该类的“execute(...)”将实际接受三个参数:

  1. 方法名称,const char* 类型
  2. 端口号,const int 类型
  3. 可选的 URL 字符串,以作为 http get header 中的 URI 进行发送

清单 8 显示了一个示例客户机,此客户机将执行“add(...)”方法,并在客户端输出结果。

清单 8. sampleClient.cpp
  1. #include <iostream>
  2. #include "XmlRpc.h"
  3. using namespace XmlRpc;
  4.  
  5. int main(int argc, char* argv[])
  6. {
  7. const char *server = "localhost";
  8. const int port = 8085;
  9. const char *uri = NULL;
  10. XmlRpcValue args, res;
  11.  
  12. XmlRpcClient c( server, port, uri);
  13. c.execute("Add", args, res);
  14. std::cout<<"result is "<<res<<std::endl;
  15. }
 

回页首

结束语

XML-RPC 是一个简单而强大的轻量级消息传递协议,可支持基于 XML 的跨异类平台通信。此标准固有的简单性在将遗留应用程序与企业集成方面表现得非常强大而实用。由于各个 XML-RPC 实现都是开放源代码的,使得此技术在企业应用程序集成领域越来越受欢迎。随着越来越多的成熟 XML-RPC 工具的出现,我们可以预见,在不久的将来,此技术将成为企业内部集成的“事实”标准。

参考资料

学习

使用 XML-RPC 为 C++ 应用程序启用 Web 服务的更多相关文章

  1. WCF 无法激活服务,因为它不支持 ASP.NET 兼容性。已为此应用程序启用了 ASP.NET 兼容性

    无法激活服务,因为它不支持 ASP.NET 兼容性.已为此应用程序启用了 ASP.NET 兼容性.请在 web.config 中关闭 ASP.NET 兼容性模式,或将 AspNetCompatibil ...

  2. .NET CORE学习笔记系列(4)——ASP.NET CORE 程序启用SSL

    一.什么是SSL? 1.概念: SSL(Secure Sockets Layer 安全套接层),及其继任者传输层安全(Transport Layer Security,TLS)是为网络通信提供安全及数 ...

  3. 什么是XML RPC?

    # -*- coding: cp936 -*- #python 27 #xiaodeng #什么是XML RPC? #中文叫:远程过程调用 #使用http协议做传输协议的rpc机制,使用xml文本的方 ...

  4. The type javax.xml.rpc.ServiceException cannot be resolved.It is indirectly

    The type javax.xml.rpc.ServiceException cannot be resolved.It is indirectly 博客分类: 解决方案_Java   问题描述:T ...

  5. WCF项目问题2-无法激活服务,因为它需要 ASP.NET 兼容性。没有未此应用程序启用 ASP.NET 兼容性。请在 web.config 中启用 ASP.NET 兼容性,或将 AspNetCompatibilityRequirementsAttribute.AspNetCompatibilityRequirementsMode 属性设置为 Required 以外的值。

    无法激活服务,因为它需要 ASP.NET 兼容性.没有未此应用程序启用 ASP.NET 兼容性.请在 web.config 中启用 ASP.NET 兼容性,或将 AspNetCompatibility ...

  6. WCF 无法激活服务,由于它不支持 ASP.NET 兼容性。已为此应用程序启用了 ASP.NET 兼容性

    作者:jiankunking 出处:http://blog.csdn.net/jiankunking 错误信息: 无法激活服务.由于它不支持 ASP.NET 兼容性.已为此应用程序启用了 ASP.NE ...

  7. IDEA 之 ERROR:无法在web.xml或使用此应用程序部署的jar文件中解析绝对uri:[http://java.sun.com/jsp/jstl/core]

    问题描述:在使用IDEA对JSTL进行测试时出现error:无法在web.xml或使用此应用程序部署的jar文件中解析绝对uri:[http://java.sun.com/jsp/jstl/core] ...

  8. Azure Front Door(三)启用 Web Application Firewall (WAF) 保护Web 应用程序,拒绝恶意攻击

    一,引言 上一篇我们利用 Azure Front Door 为后端 VM 部署提供流量的负载均衡.因为是演示实例,也没有实际的后端实例代码,只有一个 "Index.html" 的静 ...

  9. systemctl设置程序开机启动、关闭、启用/禁用服务以vsftpd为例

    参考:https://blog.csdn.net/qq_29663071/article/details/80814081 systemctl是RHEL 7 的服务管理工具中主要的工具,它融合之前se ...

随机推荐

  1. BZOJ1590:[Usaco2008 Dec]Secret Message秘密信息

    浅谈\(Trie\):https://www.cnblogs.com/AKMer/p/10444829.html 题目传送门:https://lydsy.com/JudgeOnline/problem ...

  2. fail2ban的介绍

    fail2ban的介绍 http://www.jb51.net/article/48591.htm http://lilinji.blog.51cto.com/5441000/1784726 fail ...

  3. 使用Docker快速搭建ELK环境

    今天由于Win系统的笔记本没带回家,其次Docker在非Linux系统下都需要安装额外的软件去镜像才行 所以感觉没有差别,先直接用Mac搭建一遍呢, 本篇部分命令和配置内容为摘抄 Mac下使用Dock ...

  4. 机器学习(Machine Learning)&深度学习(Deep Learning)资料汇总 (上)

    转载:http://dataunion.org/8463.html?utm_source=tuicool&utm_medium=referral <Brief History of Ma ...

  5. dubbo相关的知识点总结

    dubbo最近提交到了apache,成为了apache的孵化项目,又开始活跃起来了.就官方在git上面的说明文档和其他资料,学习总结以下dubbo的一些知识点. .The dubbo protocol ...

  6. linux下进程cpu占用过高问题定位方法

    背景 记得前段时间,同事说他们测试环境的服务器cpu使用率一直处于100%,本地又没有什么接口调用,为什么会这样?cpu使用率居高不下,自然是有某些线程一直占用着cpu资源,那又如何查看占用cpu较高 ...

  7. NFS的安装以及windows/linux挂载linux网络文件系统NFS

    1.创建linux的NFS服务端安装centos6.4,关闭防火墙/etc/init.d/iptables status yum install nfs-utils rpcbind [root@lin ...

  8. UUID是如何保证全局唯一的

    UUID由以下几部分的组合: (1)当前日期和时间,UUID的第一个部分与时间有关,如果你在生成一个UUID之后,过几秒又生成一个UUID,则第一个部分不同,其余相同.         据说时间粒度很 ...

  9. 准确计算Java中对象的大小

    由于在项目中需要大致计算一下对象的内存占用率(Hadoop中的Reduce端内存占用居高不下却又无法解释),因此深入学习了一下如何准确计算对象的大小. 使用system.gc()和java.lang. ...

  10. 安装Elastix-2.4版本

    首先,下载Elastix地址:http://www.elastix.org,下载里面的2.4版本 第一步:选择安装,Enter 选择语言,默认就行 选择us,默认 选择全部 选择默认分区,点击OK 配 ...