COM与.NET的互操作
COM与.NET的互操作中从.NET调用COM组件,如果使用VS.NET将变得非常容易,你只需要在你的工程中,添加对应的COM引用,编译工具就在后台悄悄的把COM“变成”了.NET程序集。而从传统的语言调用调用.NET组件却不如那么方便了。所以,我整理了个人调试成功的几段程序希望对大家有一些帮助,好了废话少说进入正题。
一,从vbscript等脚本调用.net组件
首先我们可以写一个.NET dll如下
//the first file:netServer.cs
using System;
using System.Reflection;
using System.Runtime.InteropServices;
[assembly: AssemblyKeyFile("key.snk")]
namespace CSharpServer
{
//缺省的是ClassInterfaceType.AutoDispatch,该方式下只生成dispatch接口
//只能被使用script、VB等late binding方式的COM客户使用
[ClassInterfaceAttribute(ClassInterfaceType.AutoDual)]
public class SharpObject
{
private string m_strName;
public SharpObject(){}
public string Name //Property: Name, Get/Set
{
get { return m_strName; }
set { m_strName = value; }
}
}
}
//the second file: test.vbs
Dim obj
Set obj = CreateObject("CSharpServer.SharpObject")
obj.Name = "Chang Ming"
MsgBox "My Name is " & obj.Name
对这两个文件按如下方式编译,为了清晰起见我们使用命令行工具(命令行工具环境可以从开始——>Microsoft Visual Studio .NET——>Visual Studio .NET 工具——>Visual Studio .NET 命令提示中进入)
1,生成密钥文件,用于给程序集强名称签名
sn -k key.snk
2,使用强名称签名,编译成类库,
csc /t:library netserver.cs
3,生成类型库
tlbexp netserver.dll /out:netserver.tlb
4,注册dll
regasm netserver.dll
5,移入gac全局程序集缓存
gacutil -i netserver.dll
6,调用测试脚本
wscript test.vbs
在这里有几个需要注意的地方,1,必须要给程序集签名,让它具有强名称。2,必须将使用regasm注册程序集,它将会在注册表中添加相应的项。3,必须将签名后的强名称程序集移入全局程序集缓存(gac)。4,必须要先安装scriptengine了,微软的脚本执行引擎。这是从脚本调用.net 程序集了,呵呵,很简单吧?J
二,从C/C++调用.NET组件
还是一段程序,呵呵,程序就是我的生命:)
//file1 name:netServer.cs
using System;
using System.Reflection;
using System.Runtime.InteropServices;
[assembly: AssemblyKeyFile("key.snk")]
namespace CSharpServer
{
public interface IObject //声明接口
{
double Sub(double c,double d);
}
//[ClassInterfaceAttribute(ClassInterfaceType.AutoDual)]
public class SharpObject:IObject
{
private string m_strName;
public SharpObject(){}
public string Name //Property: Name, Get/Set
{
get { return m_strName; }
set { m_strName = value; }
}
public double Add(double a,double b)
{
Console.WriteLine("the answer is {0}",a+b);
return a+b;
}
public double Sub(double c,double d) //实现接口方法
{
Console.WriteLine("the answer is {0}",c-d);
return c-d;
}
}
}
//file2 name: comclient.cpp
#include <windows.h>
#include <stdio.h>
#include <iostream.h>
#pragma warning (disable: 4278)
#import "netServer.tlb" no_namespace named_guids
int main(int argc, char* argv[])
{
IObject *cpi = NULL;
int retval = 1;
// Initialize COM and create an instance of the InterfaceImplementation class:
CoInitialize(NULL);
HRESULT hr = CoCreateInstance(CLSID_SharpObject,
NULL,
CLSCTX_INPROC_SERVER,
IID_IObject,
reinterpret_cast<void**>(&cpi));
if (FAILED(hr))
{
printf("Couldn''''t create the instance!... 0x%x/n", hr);
}
else
{
printf("Calling function./n");
retval = 0;
cout<<"10-4="<<cpi->Sub(10,4)<<endl;
printf("Returned from function./n");
cpi->Release();//释放com对象
}
// Be a good citizen and clean up COM:
CoUninitialize();
return retval;
}
编译方法还是如前
1,生成密钥文件,用于给程序集强名称签名
sn -k key.snk
2,使用强名称签名,编译成类库,
csc /t:library netserver.cs
3,生成类型库 //这一步很重要
tlbexp netserver.dll /out:netserver.tlb
4,注册dll
regasm netserver.dll
5,移入gac全局程序集缓存
gacutil -i netserver.dll
6,编译测试程序
cl COMClient.cpp
7,执行comclient.exe
说明:
在c/c++中调用com要麻烦一些,首先要调用COM库函数CoInitialize(NULL);进行初始化,然后调用HRESULT hr = CoCreateInstance(CLSID_SharpObject,
NULL,
CLSCTX_INPROC_SERVER,
IID_IObject,
reinterpret_cast<void**>(&cpi));
其中CLSID_SharpObject是SharpObject类(com类)的类ID它是由工具生成的用来唯一标识SharpObject类,IID_IObject唯一标识IObject接口,如果CoCreateInstance成功的创建了COM对象,那么FAILED(hr)将为false,得到com对象的指针后,就可以用它调用com对象中的方法了.