用C#编写一个进程外的COM组件示例代码讲解

简介:

代码的链接在《C#编写一个进程外的COM组件》,小技巧:如果你要同时看示例代码和讲解的话,可以用浏览器分别打开示例代码和这篇文章,然后使用Windows提供的纵向平铺窗口功能就可同时看两篇文章了。

 

TestComVisibleClass.cs里面定义了我们要发布给COM客户程序的.NET对象,由于我们的.NET进程外组件需要调用几个COM库的API,因此在ComHelperClass里面定义这些API.NET里面的声明方式,正确地声明P/Invoke函数的原型非常困难,要求程序员对Win32COM和托管代码都很熟悉才可以做,所以我写了另外一篇文章《使用Signature Tool自动生成P/Invoke调用Windows API的C#函数声明》简化P/Invoke函数声明的步骤。

 

ComHelperClass类里面CoRegisterClassObject函数的原型比较有意思,注意rclsid参数前面的[MarshalAs(UnmanagedType.LPStruct)]属性,这个属性告诉.NET,在从.NET一端传递rclsid参数值到非托管代码一端时,不要使用默认的列集(Marshal)规则,在P/Invoke里面,NET默认将结构体对象完整复制到非托管的内存里,使用UnmanagedType.LPStruct告诉NETGuid对象的指针传递给非托管函数,就省去了在调用的时候添加ref关键字的麻烦,UnmanagedType.LPStruct会有另外一篇文章来解释它,它有些特别。.NET默认将类实例对象列集成VARIANT拷贝到非托管的内存里,因此第二个参数我用[MarshalAs(UnmanagedType.IUnknown)]通知.NET需要将这个对象实例列集成IUnknow *指针。

 

我们的.NET进程外组件有可能被一些C++编写的客户端调用到,对于C++程序来说,使用前绑定(即通过接口指针调用接口成员函数)的方式会更加方便一些,否则使用延迟绑定技术(通过IDispatch接口调用接口成员函数)的方式C++代码会比较复杂一些。因此将TestComVisibleClass的一些方法和属性提取成一个接口,并且分别给ITestComVisible接口和TestComVisibleClass类分配了一个GUID,在COM世界里,前者就是我们熟知的IID,后者则是CLSID

 

同时,为了方便VB程序使用.NET进程外组件,我还特意给ITestComVisible接口的属性和函数指定了DispID,因为在OLE规范里,DispID0-1等几个特殊值的函数有特殊的意义,这一点我将会在后面的文章里讲到。

 

由于我们不能用mscoree.dll自己提供的类激活策略来在COM中激活我们的.NET对象,mscoree.dll默认提供的激活策略也会在后面的文章里讲到,我们只好显示地提供类厂,并且将我们的类厂在COM运行库里面注册一下。类厂(ClassFactory)的相关接口同样需要定义一个C#形式的原型,一个比较取巧的办法就是使用tlbimp生成一个dll,或者就是看看System.Runtime.InteropServices.ComTypes命名空间里面是否已经有定义好了的类型?IClassFactory最重要的一个函数就是CreateInstance,我们的实现就是看看客户端需要什么样子的接口,如果是IUnknown或者是我们发布的ITestComVisible102行到107行),否则就在109行的位置上抛出一个异常(碰到错误就抛异常的习惯在COM世界里不是一个友好的方式,我的代码为了简单就采取了抛异常的方式,更好的做法是返回错误码,由COM客户端决定如何处理这个错误。)

 

在程序启动的时候,我们将自己实现的类厂注册在COM运行库里面(Program.cs48行到53行),记得保留CoRegisterClassObject返回给我们的注册ID(第53行)。Program.cs里面的35行到44行可选,它们的目的是做一些安全检查,确保只有一些有权限的用户才能调用你的C# Dcom组件,如果你对安全性不关心的话,可以删除它们。在程序退出的时候(Program.cs26行到29行)扫除一些尾巴,释放一些资源。

 

那为什么我们还要一个注册表文件呢?这是因为在前绑定调用方式里,我们需要将指针在进程间传递,比如在客户端C++代码的第21行,实际上CoCreateInstanceEx需要启动我们的.NET进程(也即是进程外COM服务器),调用我们在Program.cs的第53行注册过了的IClassFactory接口来创建.NET对象实例,然后将实例的ITestComVisible接口指针从.NET进程传回C++客户端进程里来。可能有人会说,可以直接将指针的地址到C++客户端进程去嘛?这是不行的,因为Windows操作系统将进程与其它进程独立开来,简单说,一个进程里面的虚拟内存地址在另外一个进程里面可能指向一个垃圾,原理请参看操作系统书籍里面关于虚拟内存的描述。

 

为了在进程间传递ITestComVisible接口指针,COM库需要知道如何列集ITestComVisible指针,一般情况下,程序员需要提供另外一个DLL,这个DLL包含了列集ITestComVisible指针的代码。为什么要提供代码来列集指针的原因是,不同的接口包含不同的函数,例如在客户端C++代码的第25行和第26行,COM库需要列集远程函数调用(RPC),也就是需要一个方法将TestMethodRelease的调用区分开来。一般这个DLL,可以通过用msidl.exe分析IDL文件来生成列集函数的源代码编译生成。然而,这种方法比较麻烦,因此微软提供了OLEAUT32.dll,里面有一个通用的接口列集函数(但是这个函数不能列集所有的接口,可以列集的接口需要遵循一些规则,这个后面有时间再讲),可以通过分析TLB文件来列集指针,因为TLB文件相当于.NET Assembly里面的元数据,oleaut32.dll可以知道你的com组件里面有哪些接口,各个接口的声明又是怎样的,函数的参数类型是什么等等。但是oleaut32.dll需要查询注册表才能知道接口存在的Tlb文件的位置:

1.         因此注册表代码里面的第3行到第16行在注册表里面保存了类型库的存放路径,注意,在COM世界里,tlb文件也是用GUID来唯一标识的,你可以用oleview.exe打开regasm.exe或者tlbexp.exe生成的tlb文件,找到[custom(9903F14C-12CE-4c99-9986-2EE3D7D588A8)…]那一段文字来找到Tlb文件的GUID

2.         注册表代码里面的第18行到第28行,在注册表里面保存了列集ITestComVisible接口的信息,例如ProxyStubClsid指的是采用oleaut32.dll提供的通用接口列集函数,并且保存了该函数所使用tlb文件的信息(第26行到28行)。如果注册表里面没有列集接口的信息,CoCreateInstanceEx函数会返回E_NOINTERFACE(不支持此接口)错误一个让人稀里糊涂的错误代码。

3.         最后注册表代码里面的第30行到第41行向全世界声明(不好意思,不是中国人民从此站起来了的那种声明):我们的.NET对象不需要通过mscoree来在COM端激活,自己可以完成激活操作,因此我们删掉了由regasm.exe插入的InprocServer32键值,而是添加了LocalServer32键值。

 

附,上面注册表代码里面的第3行到第28行可以用RegisterTypeLib函数来完成,而RegisterTypeLib所需要的ptlib参数可以通过LoadTypeLib函数来拿到。



本文转自 donjuan 博客园博客,原文链接:http://www.cnblogs.com/killmyday/archive/2009/02/21/1395432.html   ,如需转载请自行联系原作者

相关文章
Echarts实战案例代码(19):利用visualMap视觉映射组件制作五色玫瑰工作进程图
Echarts实战案例代码(19):利用visualMap视觉映射组件制作五色玫瑰工作进程图
254 0
|
前端开发 C#
C# 基于NPOI+Office COM组件 实现20行代码在线预览文档(word,excel,pdf,txt,png)
C# 基于NPOI+Office COM组件 实现20行代码在线预览文档(word,excel,pdf,txt,png)
|
3月前
|
安全 API C#
C# 如何让程序后台进程不被Windows任务管理器强制结束
C# 如何让程序后台进程不被Windows任务管理器强制结束
87 0
|
4月前
|
SQL 网络协议 数据库连接
已解决:连接SqlServer出现 provider: Shared Memory Provider, error: 0 - 管道的另一端上无任何进程【C#连接SqlServer踩坑记录】
本文介绍了解决连接SqlServer时出现“provider: Shared Memory Provider, error: 0 - 管道的另一端上无任何进程”错误的步骤,包括更改服务器验证模式、修改sa用户设置、启用TCP/IP协议,以及检查数据库连接语句中的实例名是否正确。此外,还解释了实例名mssqlserver和sqlserver之间的区别,包括它们在默认设置、功能和用途上的差异。
|
6月前
|
SQL 开发框架 前端开发
在C#开发中使用第三方组件LambdaParser、DynamicExpresso、Z.Expressions,实现动态解析/求值字符串表达式
在C#开发中使用第三方组件LambdaParser、DynamicExpresso、Z.Expressions,实现动态解析/求值字符串表达式
|
5月前
|
存储 API C#
【Azure Developer】解决Azure Key Vault管理Storage的示例代码在中国区Azure遇见的各种认证/授权问题 - C# Example Code
【Azure Developer】解决Azure Key Vault管理Storage的示例代码在中国区Azure遇见的各种认证/授权问题 - C# Example Code
|
6月前
|
C#
C#进程调用FFmpeg操作音视频
因为公司需要对音视频做一些操作,比如说对系统用户的发音和背景视频进行合成,以及对多个音视频之间进行合成,还有就是在指定的源背景音频中按照对应的规则在视频的多少秒钟内插入一段客户发音等一些复杂的音视频操作。本篇文章主要讲解的是使用C#进程(Process)调用FFmpeg.exe进行视频合并、音频合并、音频与视频合并成视频这几个简单的音视频操作。
|
8月前
|
存储 缓存 监控
深度解析操作系统中的核心组件:进程管理与内存优化
【5月更文挑战第29天】 在现代计算技术的心脏,操作系统扮演着至关重要的角色。它不仅管理和控制计算机硬件资源,还为应用程序提供了一个运行环境。本文将深入探讨操作系统中的两个核心组件——进程管理和内存管理,并分析它们对系统性能的影响以及如何通过技术手段实现优化。通过对操作系统内部机制的剖析,我们将揭示这些组件是如何相互作用,以及它们如何共同提升系统的响应速度和稳定性。
|
8月前
|
编解码 C#
C#使用PPT组件的CreateVideo方法生成视频
C#使用PPT组件的CreateVideo方法生成视频