要求:
为一个WinForm程序暴露一个COM接口,让其它应用程序能够以COM服务器(LocalServer)方式启动这个程序并且对其进行操作
如果发现已经在运行的应用程序,则直接重用当前运行的应用程序进行操作。

分析:
根据要求,分解具体需要解决的技术问题如下:
在WinForm程序中定义一个COM visible接口并实现。
将这个Winform程序变为COM服务器(LocalServer)。
将Winform程序的COM对象加入系统的ROT表中。

验证方法:
客户端通过CoCreateInstance(LocalServer)方式激活这个COM对象,应当看到对应的WinForm程序启动,并且CoCreateInstance成功返回我们所需的Interface指针,
客户端调用Interface的相关方法,Winform程序能够成功执行。
Winform程序运行时,客户端能够在ROT取得Winform程序的IUnknown指针。能够成功QI 成所实现的COM接口,并且调用相关方法成功执行。

解决方案:
1. 在WinForm程序中定义一个COM visible接口并实现
在.Net中定义COM 接口可以通过在接口定义上添加GuidAttribute和InterfaceTypeAttribute,定义该接口的IID并告知CLR该接口需要同时导出为普通的IUnknown COM 接口和OLE automation接口。具体例子如下:

 

.Net下进程外COM服务器的实现[InterfaceType(ComInterfaceType.InterfaceIsDual)]
.Net下进程外COM服务器的实现[Guid(
"CF7C704A-6AC3-4963-8818-EF1493CEC2D1")]
.Net下进程外COM服务器的实现
public interface IProvider
}

实现这个interface,

 

.Net下进程外COM服务器的实现[ClassInterface(ClassInterfaceType.None)]
.Net下进程外COM服务器的实现[Guid(
"58C142C7-E599-4921-BF29-33DC0FCCBECA")]
.Net下进程外COM服务器的实现
public class ProviderImp : IProvider
 

 

这部分和.Net中实现进程内COM服务器是相同的,在.Net Framework SDK的文档中有详细的介绍。关于在.Net中实现COM组件可以参看MSDN。

2. 将这个Winform程序变为COM服务器(LocalServer)
根据COM本质论中的论述实现进程外COM服务器的需要以下几方面条件;
注册表中对应的CLSID下需要添加LocalServer32键,并把default设为EXE程序的路径
进程外服务器需要在启动时主动向SCM(Service Control Manager)中注册COM Class Object,这样SCM才能创建出对应的COM object返回给客户端,因此.Net程序需要提供一个Class Object(一个实现了IClassFactory COM接口的对象)并调用CoRegisterClassObject将其注册到SCM中。
除此之外,.Net中需要使用regasm命令将assembly中的COM visible类型加入注册表。(注意如果assembly没有加入GAC,请在注册的时候加上/codebase参数否则.Net会无法加载对应的assembly产生奇怪的E_NOINTERFACE错误)

具体的实现方式根据使用的.Net版本有所差异:对于.Net v2.0及其后版本而言,.Net RegistrationServices类提供了RegisterTypeForComClients和UnregisterTypeForComClients方法能够帮助我们很方便的实现注册和注销。

 

.Net下进程外COM服务器的实现private static int cookie;
.Net下进程外COM服务器的实现
private static RegistrationServices msRegSvc = new RegistrationServices();
.Net下进程外COM服务器的实现
public static void RegisterServer()

对于.Net 2.0之前的情况,我们可以通过下面的方式使用.Net自己的IClassFactory实现来实现SCM的注册

 

.Net下进程外COM服务器的实现private static uint appId = 0;
.Net下进程外COM服务器的实现
private static void RegisterServerImp(Guid clsid)

3. 将Winform程序的COM对象加入系统的ROT表中

加入ROT表有很多办法,可以通过取得系统IRunningObjectTable接口来直接注册也可以使用OLE的API RegisterActiveObject来实现,这里由于没有特殊要求,我们使用后者来实现。

 

.Net下进程外COM服务器的实现//define static object to keep COM object alive in whole application lifecycle.
.Net下进程外COM服务器的实现
private static ProviderImp msProvider = new ProviderImp();
.Net下进程外COM服务器的实现
private static int dwRegister = 0;
.Net下进程外COM服务器的实现
public static void RegisterActiveObject()

附录:
1.相关API的PInvoke定义:

.Net下进程外COM服务器的实现[DllImport("Ole32.Dll")]
.Net下进程外COM服务器的实现
public static extern int CreateBindCtx(int reserved,out IBindCtx bindCtx); 
.Net下进程外COM服务器的实现
.Net下进程外COM服务器的实现[DllImport(
"oleaut32.dll")]
.Net下进程外COM服务器的实现
private static extern int RegisterActiveObject([MarshalAs(UnmanagedType.IUnknown)] object pUnk, ref Guid rclsid, uint dwFlags, out int pdwRegister); 
.Net下进程外COM服务器的实现
.Net下进程外COM服务器的实现[DllImport(
"oleaut32.dll")]
.Net下进程外COM服务器的实现
private static extern uint GetActiveObject(ref Guid rclsid, IntPtr pvReserved, [Out]out IntPtr ppunk); 
.Net下进程外COM服务器的实现
.Net下进程外COM服务器的实现[DllImport(
"oleaut32.dll")]
.Net下进程外COM服务器的实现
private static extern uint RevokeActiveObject(int dwRegister, IntPtr lpReserved); 
.Net下进程外COM服务器的实现
.Net下进程外COM服务器的实现
//from www.pinvoke.net

2.C++客户端测试代码

 

.Net下进程外COM服务器的实现#include "stdafx.h"
.Net下进程外COM服务器的实现#import 
"Server.tlb" 
.Net下进程外COM服务器的实现
.Net下进程外COM服务器的实现
class COMHelper

相关文章:

  • 2022-12-23
  • 2022-12-23
  • 2021-11-05
  • 2022-12-23
  • 2021-08-07
  • 2022-12-23
  • 2021-12-02
  • 2021-05-15
猜你喜欢
  • 2021-06-08
  • 2021-06-23
  • 2021-08-24
  • 2021-06-11
  • 2021-08-20
  • 2022-12-23
相关资源
相似解决方案