当我们运行一个Windows服务的时候,一般情况下,我们会选择以非窗口或者非控制台的方式运行,这样,它就只是一个后台程序,没有界面供我们进行交互。

那么当我们想与Windows服务进行实时交互的时候,我们应该怎么做呢?

快速给Windows服务添加实时交互功能的方案

Windows服务是一个进程,而我们用于交互的程序,又是另外一个进程。我们与Windows服务实时交互,其实就是一个进程间通信的问题。所有的进程间通信的方案基本上都适用于实时交互的方案,比如Socket、共享内存、管道、COM等。

这些方案中,当属COM的开发最快速,因为我们是给基于ATL的Windows服务添加COM接口嘛。

COM简介

组件对象模型,英文为Component Object Model,缩写COM,是微软的一套软件组件的二进制接口标准。这使得跨编程语言的进程间通信、动态对象创建成为可能。COM是多项微软技术与框架的基础,包括OLE, OLE自动化, ActiveX, COM+, DCOM, Windows shell, DirectX, Windows Runtime。详细介绍可以参考 组件对象模型

给服务添加COM接口

创建基于ATL的Windows服务可以参考 玩转Windows服务系列——创建Windows服务

接下来,快速给服务添加COM接口。

首先给项目添加了一个ATL简单对象,如下:

玩转Windows服务系列——给Windows服务添加COM接口-LMLPHP

玩转Windows服务系列——给Windows服务添加COM接口-LMLPHP

玩转Windows服务系列——给Windows服务添加COM接口-LMLPHP

按上面步骤创建了ATL简单对象后,会产生这么一个文件:

ServiceComTest.idl

文件内容如下:

import "oaidl.idl";
import "ocidl.idl"; [
object,
uuid(4DDE5CA3-F5D7-4BC3-9045-E697297C5530),
dual,
nonextensible,
pointer_default(unique)
]
interface IIServiceComTest : IDispatch{
};
[
uuid(54A347BA-7689-4578-A346-C96D924BD637),
version(1.0),
]
library ServiceComTestLib
{
importlib("stdole2.tlb");
[
uuid(C264868C-91E7-4BFE-8DD9-32D0804E44F6)
]
coclass IServiceComTest
{
[default] interface IIServiceComTest;
};
};

这个idl文件就是用来定义COM接口的。

接下来给接口添加新的方法。

在类视图中,找到刚刚生成的接口 IIServiceComTest:

玩转Windows服务系列——给Windows服务添加COM接口-LMLPHP

然后右键菜单,添加方法:

玩转Windows服务系列——给Windows服务添加COM接口-LMLPHP

玩转Windows服务系列——给Windows服务添加COM接口-LMLPHP

玩转Windows服务系列——给Windows服务添加COM接口-LMLPHP

这样,就添加了一个add方法,x、y为输入,result为输出。

然后可以在idl文件中看到add方法的定义:

interface IIServiceComTest : IDispatch{
[id(1), helpstring("两个整数相加")] HRESULT add([in] LONG x, [in] LONG y, [out, retval] LONG* result);
};

实现COM接口

我们给COM接口添加的方法,只是一个声明、描述,我们还必须实现这个方法,其他进程才能与此服务通信。

在IServiceComTest.cpp文件中可以找到此方法:

STDMETHODIMP CIServiceComTest::add(LONG x, LONG y, LONG* result)
{
// TODO: 在此添加实现代码 return S_OK;
}

接下来就是实现此方法,如下:

STDMETHODIMP CIServiceComTest::add(LONG x, LONG y, LONG* result)
{
*result = x + y;
return S_OK;
}

这样,一个完整的COM接口及其实现就算是完成了,接下来需要通过测试程序调用此接口进行测试了。

调用COM接口

创建一个基本的控制台程序,然后将初始化测试代码,进行测试,代码如下:

#include "..\ServiceComTest\ServiceComTest_i.c"
#include "..\ServiceComTest\ServiceComTest_i.h"
#include <iostream>
using namespace std; int _tmain(int argc, _TCHAR* argv[])
{
IIServiceComTest* test;
CoInitialize(NULL);
auto hresult = CoCreateInstance(CLSID_IServiceComTest,
NULL,
CLSCTX_LOCAL_SERVER | CLSCTX_INPROC_HANDLER,
IID_IIServiceComTest,
(void**)&test); LONG x = 1;
LONG y = 2;
LONG result = 0;
hresult = test->add(x, y, &result);
cout << "result is " << result << endl;
system("pause");
}

这里,只是一个演示程序,省略了代码的错误处理。

运行程序,得到了正确的结果,result is 3, 结果如下:

玩转Windows服务系列&mdash;&mdash;给Windows服务添加COM接口-LMLPHP

参考资料

Step by Step COM Tutorial

COM in C++

COM(C++) programming tutorials

C/C++ COM Code Example: Reading Messages Asynchronously

系列链接

玩转Windows服务系列——创建Windows服务

玩转Windows服务系列——Debug、Release版本的注册和卸载,及其原理

玩转Windows服务系列——无COM接口Windows服务启动失败原因及解决方案

玩转Windows服务系列——服务运行、停止流程浅析

玩转Windows服务系列——Windows服务小技巧

玩转Windows服务系列——命令行管理Windows服务

玩转Windows服务系列——Windows服务启动超时时间

玩转Windows服务系列——使用Boost.Application快速构建Windows服务

玩转Windows服务系列——给Windows服务添加COM接口

04-28 06:47