由于C#编写的是托管代码,编译生成微软中间语言,而C++代码则编译生成本地机器码(这种C++也有叫做本地C++或者非托管C++,VC6.0就是用于开发非托管C++代码的平台),这两种语言进行混合编程就存在一定困难。比较常用的方法是使用DllImport的方法,这种方法在网上有很多介绍,这里就不详细叙述了。但是用过这种方法的人都知道这种方法对于导出函数还可以但是却没法导出非托管C++类!非常的要命。
然而,除了C#、非托管C++外,C系列中还存在一种语言叫做托管C++,这种语言语法上和非托管C++几乎一样,但是却和C#一样编译成为微软中间语言,这样托管C++就可以和C#良好地通信,即可以在C#中使用托管C++类。另外,托管C++还有两个及其重要的特性就是:可以调用非托管C++的类和函数!托管C++的程序集可以嵌套非托管C++编译的机器码!好强大的混合体。所以我们的技术路径也就明晰了:C#以托管C++为中介调用非托管C++的类和函数。换句话说也就是用托管C++给非托管C++代码做一个外壳包装供C#调用。
环境:VS2017,Win10;
第一步:新建CLR C++类库。将其命名为Test_DLL
第二步:新建非托管方法 FunctionAdd类
FunctionAdd.h:
//这里定义用于导出的C函数
int Add(int a, int b);
FunctionAdd.cpp:
#include "stdafx.h"
#include "FunctionAdd.h"
//这里实现C函数(算术加法)
int Add(int a, int b)
{
return a + b;
}
第三步:新建非托管类Native类
Native.h:
#pragma once
class Native
{
public:
Native(void);
~Native(void);
int menber;//用于导出的成员
int menderFuncSub(int a, int b);//用于导出的成员函数,实现算术减法
};
Native.cpp:
#include "stdafx.h"
#include "Native.h"
Native::Native(void)
{
//构造
menber = 1;
}
Native::~Native(void)
{
}
//这是非托管C++类实现的算术减法
int Native::menderFuncSub(int a, int b)
{
return a - b;
}
第三步 新建托管类clrClass
clrClass.h:
#pragma once
#include "Native.h"
//这是个托管C++类用于对C++本地代码类和函数等进行封装以在C#中使用
public ref class clrClass
//必须声明为public,否则类在程序集中不可见,关键词ref表示类是一个托管类,将编译为中间语言
{
public:
clrClass(void);
int menber;//这个成员访问非托管类CClassNative的公共成员(事实上只需要包装公共成员和公共成员函数,私有的包装没有意义,也包装不了)
int menderFuncSub(int a, int b);//这个成员函数用于包装非托管类CClassNative的公共成员函数
int menberFuncAdd(int a, int b);//这成员函数用于包装C函数 int Add(int a,int b)
private:
Native * native;//建立一个非托管类实例 (在构造函数中实例化,这里可以理解为是为了在clrClass中“继承”CClassNative的公共成员和方法)
};
clrClass.cpp:
#include "stdafx.h"
#include "clrClass.h"
#include "FunctionAdd.h"
using namespace System;
clrClass::clrClass(void)
{
native = new Native();//这里一定注意要创建对象!
menber = native->menber;//这里是简单举例,最好用属性方法来读写CClassNative类的成员,同C#相似,托管C++中有属性函数,用法请自己查
}
//通过调用非托管类CClassNative的减法函数实现算术减法
int clrClass::menderFuncSub(int a, int b)
{
return native->menderFuncSub(a, b);
}
//通过调用C函数实现算术加法
int clrClass::menberFuncAdd(int a, int b)
{
return Add(a, b);
}
这样就实现了clrClass对Native类和C函数int Add(inta,int b)的托管包装。生成的Test_DLL.dll”就可以直接在C#中用了
第四步:新建WinForm程序,然后引用Test_DLL.dll
代码:
public partial class Form1 : Form
{
clrClass clr = null;
public Form1()
{
InitializeComponent();
clr = new clrClass();
}
private void button1_Click(object sender, EventArgs e)
{
int a = Convert.ToInt32(txtA.Text);
int b = Convert.ToInt32(txtB.Text);
//通过托管C++调用C函数int Add(int a,int b)实现的加法
txtC.Text = clr.menderFuncSub(a, b).ToString();
}
private void btnAdd_Click(object sender, EventArgs e)
{
int a = Convert.ToInt32(txtA.Text);
int b = Convert.ToInt32(txtB.Text);
//通过托管C++调用C函数int Add(int a,int b)实现的加法
txtC.Text = clr.menberFuncAdd(a, b).ToString();
}
}