这个主题在《c#高级编程》中有提到,可以下本电子书来看看。一下我也列出自己的学习笔记。
根据网上的说法,C#调用C++至少有以下几种方法:
1通过COM组件调用
2通过MC++转接
3利用Intptr
由于我只能成功使用COM组件的方法,所以下面只谈一下这个方法。
一.通过ALT建立COM组件
1、创建ATL Project
步骤1:建立一个解决方案。 步骤2:在 该解决方案中,新建一个 vc++ 的 ATL 项目。见下图
选择Dynamic-link library(DLL)
2、向工程中添加ATL对象
选择ATL Simple Object
名字为FunTest
Interface选择Custom
3、在接口中添加方法.(在类视图中)
方法名为Sum,参数如下图,有三个参数
接口方法的实现, 这个太简单了
STDMETHODIMP CFunTest::Sum(LONG num1, LONG num2, LONG* pVal) { // TODO: Add your implementation code here *pVal = num1 + num2; return S_OK; }
二.在C#中调用COM组件
1、新建一个C# console项目,比如叫Test(winform的也是一样道理,只是Console调试更方便而已。第一次在C#中调用COM组件时,建议先用Console试试,否则会搞得一头烟)
2、添加引用:
在弹出的对话框中选COM子页,找到ATLProjectTest 1.0类型库(或类似的东西)
就会发现在Test->引用下多了ATLProjectTestLib
3在主函数(program.cs)中调用组件
using System; using System.Collections.Generic; using System.Linq; using System.Text; using ATLProjectTestLib; namespace Test { class Program { static void Main(string[] args) { ATLProjectTestLib.FunTestClass f1 = new FunTestClass(); Console.Write (f1.Sum(3,4)); } } }
三.很多的细节及注意事项
仅按上面一步步地操作下来发现还会出错,会说找不到XX COM组件,或是COM组件没注册。实际上很多不当操作都会导致上面这个错误。只有当你生成的文件中有Interop.ATLProjectTestLib.dll时,你才算成功了。
1、所有修改都要重新编译。如修改了ATL项目,首先要重新编译ATL,再重新编译C# project。当把项目移到另一台计算机,有时编译器由于ATL项目没有被修改,而不会重新编译,这就无法写注册表。这时可以把ATL项目的Debug文件夹删除,再重新编译。
2、在创建ATL project时,由于一个选Dual或Custom的地方,我认为应该选Dual,但网上较多文章选Custom。
以下是选了Custom后应做的补救,
回到ATLProjectTest,在ATLProjectTest.idl中(或双击 类视图->IFunTest)
import "oaidl.idl"; import "ocidl.idl"; [ object, uuid(90477521-7218-4CA1-AADC-EAF3C42AB24B), dual, //--------按原做法不设这个属性,只有设了这个属性才不会错 helpstring("IFunTest 接口"), pointer_default(unique) ]
3、ATL默认按32位编译,而C#中一般是按any CPU形式,像我是64位机的就要小心了。要改一下C#项目中的生成规则。
点击项目的属性页:
设置目标平台为x86
4、做完上面这3点,大概Test应该能运行了。可以先跳到第四大点看完后再倒回来。
在实际的工程中,一般有很多个项目,假设以C#项目为主,还有一个ATL项目ATLProjectTest。而这个ATLProjectTest被Business项目(逻辑层)调用,而Business又被Form项目(用户层)调用,这都是很常见的。
在“编译”时就不能直接启动调试,要按ATLProjectTest->Business->Form的顺序一个个地生成项目。
5、有时会因为找不到Interop.ATLProjectTestLib.dll而出错,可以试着把各个项目的输出路径都改到同一个文件夹(如用户层工程的文件夹)。
四.更具体的编程
ATL与C#类型对应
ATL |
C# |
LONG |
int |
DOUBLE |
double |
CHAR |
sbyte |
VARIANT_BOOL |
bool |
BSTR |
string |
* |
ref |
在C++中更多用STL string来写
BSTR与STL string的转化方法
#include <iostream> #include <string> #include<comutil.h> #pragma comment(lib,"comsuppw") using namespace std; string str; //STL string BSTR bs; bs = _com_util::ConvertStringToBSTR(str.c_str()); str = _com_util::ConvertBSTRToString(bs);
解决了字符串的问题数组和对象都好办,因为可以序列化。
有关构造函数和析构函数我还不太会怎么做,但可以自定义Init()和ToEnd()来应付一下。
就写到这里了,还有很多不懂得,请多多指正!