最近研究了一下网络通信中间件ICE的使用,粗通其皮毛,按照官方手册依葫芦画瓢写了一个程序员都喜闻乐见的“Hello World”程序,服务端和客户端均用C++开发,通讯协议使用默认的TCP。感觉ICE的大致好处有以下两点:
- 平台无关性。无论客户端或者服务端均可用现在流行的开发语言(C++ /JAVA/C#/php)进行开发,并且屏蔽语言差异性。现在比较流行的方式是客户端用C#开发,与用C++开发的服务端直接通讯。
- 通讯协议多样性。现在可选择TCP、UDP、HTTP进行通讯,如果对安全要求较高,可选择SSL对传输的数据进行加密。
另外ICE还提供一些扩展组件,实现网络通讯的负载均衡(ICEGrid),通讯节点统一管理(ICEBOX),程序自动更新(ICEPatch)等,方便应用扩展。
如果使用过ICE的人都知道,在实现服务端或者客户端的时候通常都要写一些“公式化”的代码,负责Ice通信器初始化、异常捕获,以及应用终止后的销毁。如下所示:
1 int status=0;
2 Ice::CommunicatorPtr ic;
3 try
4 {
5 ic = Ice::initialize(argc,argv);
6 Ice::ObjectAdapterPtr adapter =
7 ic->createObjectAdapterWithEndpoints("SayHelloAdapter","tcp -h 127.0.0.1 -p 10000");
8 Ice::ObjectPtr object = new HelloICEI;
9 adapter->add(object,ic->stringToIdentity("SimpleHello"));
10 adapter->activate();
11 ic->waitForShutdown();
12 }
13 catch (const Ice::Exception & e)
14 {
15 cerr << e << endl;
16 status = 1;
17 }
18 catch (const char * msg)
19 {
20 cerr << msg << endl;
21 status = 1;
22 }
23 if ( ic )
24 {
25 try
26 {
27 ic->destroy();//关闭ICE
28 }
29 catch (const Ice::Exception & e)
30 {
31 cerr << e << endl;
32 status = 1;
33 }
34 }
如果每次都要写这么多的话,第一浪费时间和精力,第二不能将注意力集中到业务逻辑上。还有一点就是对代码的整洁和易读方面做的不够好(本人略微有些代码“洁癖”)。
阅读官方手册,发现ICE提供两个工具类封装了这些“公式化”逻辑,分别是“Application”和“Service”。
Ice::Application本身是一个抽象类,其run()函数为纯虚函数,因此必须被继承后使用。
Ice::Application 是一个单体(singleton)类,会创建单个通信器。如果你要使用多个通信器,不能使用Ice::Application来定义多个App。而至多定义一个App的实例。
其它通信器需要使用Ice::initialize()手工生成。
一般而言,Ice::Application 类对于Ice 客户和服务器来说已经非常方便,但在有些情况下,应用可能需要作为Unix 看守(daemon)或Win32 服务运行在系统一级。对于这样的情况,Ice 提供了Ice::Service。一个可与Ice::Application 相比的单体类,但它还封装了低级的、 针对特定平台的初始化和关闭步骤――系统服务常常需要使用这样的步骤。
下面介绍Ice::Service的使用方法,仅介绍开发步骤,后面附上具体例子下载。
1. 需要在服务端中引用Ice/Service.h头文件。本人曾在此浪费了2个小时。
2. 新建一个类继承Ice::Service,并实现其中的三个虚函数。代码如下:
在main函数中启动服务。此处需要注意对于服务来说最常见的有安装服务、卸载服务、启动服务、停止服务四个操作。这四个操作分别通过启动参数来控制。以Win32平台为例:
1 class MyService : public Ice::Service
2 {
3 protected:
4 virtual bool start(int, char *[],int&);
5 virtual bool stop();
6 virtual void interrupt();
7 private:
8 Ice::ObjectAdapterPtr m_adapter;
9 };
10 void MyService::interrupt()
11 {
12 std::cout << "Receive signal ..." << std::endl;
13 Ice::Service::interrupt();
14 }
15 bool MyService::stop()
16 {
17 std::cout << "Stop running ..." << std::endl;
18 return true;
19 }
20 bool MyService::start(int argc, char * argv[],int& status)
21 {
22 std::string endpoint = "tcp -h localhost -p 10000";
23 m_adapter = communicator()->createObjectAdapterWi thEndpoints("SimpleHelloAdapter", endpoint);
24 Ice::ObjectPtr object = new HelloICEI;
25 m_adapter->add(object,communicator()->stringToIdentity("SimpleHello"));
26 m_adapter->activate();
27 return true;
28 }
--service NAME
作为名叫NAME 的Windows 服务启动。在传给start 成员函数的参数向量中,这个选项会被移除。
但是,在应用作为Windows 服务运行之前,它必须先被安装,因此,Ice::Service 类还支持另外一些的命令行选项,用于执行管理活动:
--install NAME [--display DISP] [--executable EXEC][ARG ...]
安装NAME 服务。如果指定了--display 选项,就把DISP 用作服务的显示名,否则就使用NAME。如果指定了--executable 选项,就把EXEC 用作服务的可执行路径名,否则就使用可执行文件的路径名来调用--install。其他任何参数都会不加改变地传给Service::start 成员函数。注意,在启动时传给服务的参数集中,这个命令会自动增加命令行参数--service NAME,因此,你不需要显式地指定这些选项。
--uninstall NAME
移除NAME 服务。如果服务目前是活动的,在反安装之前,必须先使它停止。
--start NAME [ARG ...]
启动NAME 服务。其他任何参数都会不加改变地传给Service::start 成员函数。
--stop NAME
停止NAME 服务。如果指定的管理命令不止一个,或者在使用--service 的同时还使用了管理命令,就会发生错误。在执行了管理命令之后,程序会立即终止。Ice::Service 类支持Windows 服务控制代码SERVICE_CONTROL_INTERROGATE 和SERVICE_CONTROL_STOP。在收到SERVICE_CONTROL_STOP 时,Ice::Service 会调用shutdown 成员函数。
在Visual Studio中设置启动参数—install MyService,可在系统服务中注册名为“MyService”的服务。
启动服务端程序可以在系统服务列表中看到MyService已经作为服务注册进去了。
4. 至此ICE的服务端即可作为系统服务的形式存在与客户端进行通讯了。