C#调用C++ DLL 中定义的函数

来源:互联网 发布:淘宝客教学视频 编辑:程序博客网 时间:2024/06/05 05:39

公司的系统主要是用c#实现的,现在要用到其他公司编写的接口,用c++实现,因此涉及到c#调用c++的部分,已成功调用,现用个简单的例子记录实现过程

创建c++ dll

struct  Person{char name[5];int age;};extern "C" __declspec(dllexport) Person* Test(){Person* p= new Person;strcpy_s(p->name, "beck");p->age = 32;return p;}

将此代码编译为dll, vs 2013中, 右键项目->属性,常规中将目标文件扩展名和配置类型改为"dll"

编译好后将生成的dll复制到c#的运行目录下


c#程序调用dll

定义c#中和c++对应的结构体

 [StructLayout(LayoutKind.Sequential, CharSet = CharSet.Ansi)]    struct Person    {        [MarshalAs(UnmanagedType.ByValTStr, SizeConst = 5)]        public string name;        public int age;    }

属性
[StructLayout(LayoutKind.Sequential, CharSet = CharSet.Ansi)]
表示此结构体在内存中是顺序布局,编码为ansi

定义外部方法

       [DllImport("InvokeStructSample.dll")]        public static extern IntPtr Test();

注意返回类型为IntPtr,是一个指针, 而非直接的Person结构体, 这里我卡了很久,是因为对两边机制的理解不深导致。 如果直接返回Person, Person中纸包含简单类型(int,doule之类的)不会有问题,但如果包含复杂类型(string, 数组之类的) 则会报 "方法的签名与pinvoke不兼容" 的错误, 之前一直在c#结构体的属性上找出路,没成功,故而改为指针的形式。

调用:

            IntPtr pPerson = Test();            Person p = (Person) Marshal.PtrToStructure(pPerson, typeof(Person));            Console.WriteLine(p.name);            Console.WriteLine(p.age);

执行,调用成功!!


另外,如果函数中需要传递结构体, c++中传结构体的指针, c#中是将参数定义为IntPtr,然后调用时用StructureToPtr()方法将结构体转成IntPtr,传入调用。



这个问题折腾了我一天,终于解决了,这样就可以让c#顺利调用用c++实现的函数了。希望能对同样困扰于此的小伙伴有所帮助

1 0
原创粉丝点击