1200字范文,内容丰富有趣,写作的好帮手!
1200字范文 > C#与VC相互调用之C#调用VC的DLL

C#与VC相互调用之C#调用VC的DLL

时间:2019-09-04 18:53:36

相关推荐

C#与VC相互调用之C#调用VC的DLL

介绍

每种语言都有其优势和特点,复杂的项目一般需要2种及以上的语言,这样才能发挥每种语言的优势。这里就介绍一下C#如何调用VC的DLL

项目创建

打开VS,因为涉及到两个程序,这里先创建一个空的解决方案:

然后再来创建一个VC的DLL项目:

选择Win32项目模板:

选择应用程序的类型为DLL:

这里不用勾选“导出符号”,稍后自己来写导出就行。点击完成后生成如下文件:

dllmain.cpp文件不用修改,只在Win32Dll.cpp中添加我们用来测试的函数即可:

就简单写了个加减函数,下面再创建一个C#项目,来测试DLL的调用:

这里只是简单测试VC的DLL,并不需要界面,所以创建一个控制台应用程序即可,文件结构如下:

为了导入DLL的函数,再添加一个C#类:

在生成的TestDll类中导入VC库定义的函数声明:

这里注意两个地方,库函数名字就是VC项目生成的库的名字(注意别写错),调用方式要选择“CallingConvention.Cdecl”。而看到DllImport画红线报错了,需要添加“InteropServices”引用:

导入库函数后,就可以在main函数中编写调用代码了:

最后一行ReadLine()是防止控制台程序一闪而逝的,到这里测试代码都写完了。先别着急编译,因为C#程序要调用DLL必须保证生成的exe和dll文件在同一目录,所以需要改一下VC程序生成dll的路径:

两个项目在一个解决方案下,所以前面路径是$(SolutionDir)\,C#程序默认生成路径是项目文件夹的bin目录。又因为解决方案下有多个项目,还要把C#程序设置为启动项目:

设置好后就可以编译运行了:

正常情况就顺利出结果了,如果报错,可能就是哪个路径设置不对,再仔细核对下exe和dll是否都生成在同一目录了:

字符串参数传递

上面只是最简单的调用测试,实际传参比较麻烦的就是字符串了,由于C#操作的是数组,而VC使用指针传参,下面就看看这俩之间如何转化。先写VC库中函数:

这里写了个输入输出函数,根据输入不同返回不同的字符串,下面看看C#里如何调用,同样TestDll类中先导入声明:

在C#里面指针参数都改为数组了,但前面有In或Out属性来区分。再来写main里面的调用:

这样写好后运行一下:

结果显示是没问题,但调试看str的值:

字符串变量后面跟这么多\0总不好,这里需要处理一下:

加一行代码就搞定,C#就是这么方便。那位说了,C#不是也能使用指针吗,有人就是不习惯使用数组。那好,我们再来写一个传指针的函数:

由于使用了std命名空间,需要加上头文件:

#include <iostream>

注意看一下TestDll类中声明与之前有什么不同:

这里由于参数是指针,所以要加上unsafe修饰符,这样编译才能顺利进行。这里有个需要注意的地方,C#定义的参数是char,为什么VC那边是wchar_t呢?熟悉C#应该知道,其使用的是Unicode编码,char是16位而非VC里面的8位,因此VC必须用对应的宽字符变量。再看看调用代码:

unsafe修饰符报错了,根据提示,得在项目属性里面设置“允许不安全代码”:

这样编译就没问题了:

当然这里只是为了指针参数测试,一般不推荐这种方式,因为传递的指针空间大小另一边无法检测,容易造成内存溢出。

传递结构体参数

下面再来个复杂点的,如果VC的库里面有类定义,那么C#如何调用呢?首先VC创建一个UserInfo类,添加头文件:

敲入类代码:

然后写这个类的创建函数:

不要直接传参类变量,所以这里又定义一个User结构体作为参数进行传递,那么C#那边如何声明呢:

同样定义了一个User结构体,而Create函数返回值只是一个IntPtr指针,调用的时候再转换为对应的结构体:

由于用到Marshal类,所以需要引入命名空间:

using System.Runtime.InteropServices;

再次编译运行:

细心读者可能注意到VC里面的User变量是malloc出来的,使用Marshal.PtrToStructure传给C#的时候是不是就托管了,以后自动释放呢?答案是否定的,因为Marshal.PtrToStructure会进行内存拷贝的,C#只负责释放拷贝后的自己管理的变量,而原来VC里的User还需要我们自己释放。为此编写释放代码测试下:

C#中声明该函数:

调用测试:

我们看到,在参数传递之后就释放了VC库里User变量,如果没进行内存拷贝,下面的输出调用肯定要报错的,可实际运行仍然显示正常,说明Marshal.PtrToStructure完成了内存拷贝。关于这一点使用时要非常注意,C#的内存自动管理,而VC库的内存还是需要自己管理,否则容易造成内存泄露。

调用MFC的DLL

上面讲的都是调用Win32的DLL,那么C#能不能也调用MFC的DLL呢?下面来测试一下,再往解决方案里添加个MFC的项目:

按照默认来创建即可,然后添加一个MFC类:

填好类名和基类:

点击完成,然后设计一个简单的对话框界面:

然后在MFCDll.cpp文件中编写导出函数:

在C#程序中声明该函数:

注意DLL的名称要与新建的MFC项目生成的名称一致,这个没有传参,直接在main函数中调用即可:

TestDll.ShowDlg();Console.ReadLine();

同样别忘设置MFCDll项目的生成路径为C#的bin路径:

编译后确保3个项目都生成成功,然后运行:

好了,先写到这里吧,下一篇再介绍VC调用C#的DLL。

本内容不代表本网观点和政治立场,如有侵犯你的权益请联系我们处理。
网友评论
网友评论仅供其表达个人看法,并不表明网站立场。