欢迎来到尧图网

客户服务 关于我们

您的位置:首页 > 新闻 > 社会 > C#与C++互操作

C#与C++互操作

2024/12/25 14:42:10 来源:https://blog.csdn.net/zhaotianff/article/details/141207634  浏览:    关键词:C#与C++互操作

一、C#调用C++库

1、创建C++库


打开VisualStudio,创建一个C++工程,输入项目名称HelloWorldLib

确定,然后下一步。选择应用程序类型为DLL

单击完成,我们就创建好了一个C++库的项目。

这里为了方便,我们直接在HelloWorldLib.cpp里定义函数

C++库导出有两种方式

一、以C语言接口的方式导出

这种方法就是在函数前面加上 extern "C" __declspec(dllexport)

加上extern "C"后,会指示编译器这部分代码按C语言的进行编译,而不是C++的。

 1 #include "stdafx.h"2 #include<iostream>3 4 extern "C" __declspec(dllexport) void HelloWorld(char* name);5 6 7 extern "C" __declspec(dllexport) void HelloWorld(char* name)8 {9     std::cout << "Hello World " << name << std::endl;
10 }

二、以模块定义文件的方式导出

在工程上右键,选择添加-》新建项

然后选择代码-》模块定义文件

在Source.def中输入

LIBRARYEXPORTS
HelloWorld

EXPORTS下面就是要导出的函数,这里不需要添加分号隔开,直接换行就行。

此时,我们函数的定义如下

 1 #include "stdafx.h"2 #include<iostream>3 4 void HelloWorld(char* name);5 6 7 void HelloWorld(char* name)8 {9     std::cout <<"Hello World "<< name << std::endl;
10 }

编译,生成dll。这里需要注意的是,如果生成是64位的库,C#程序也要是64位的,否则会报错。

2、使用C#调用

接下来我们新建一个C#控制台项目

打开前面C++库生成的目录,将HelloWorldLib.dll复制到C#工程的Debug目录下。也可以不复制,只需在引用dll的时候写上完整路径就行了。这里我是直接复制到Debug目录下

 1 using System.Runtime.InteropServices;2 3 namespace ConsoleApplication24 {5     class Program6     {7         [DllImport("HelloWorldLib.dll")]8         public static extern void HelloWorld(string name);9 
10         //可以通过EntryPoint特性指定函数入口,然后为函数定义别名
11 
12         [DllImport("HelloWorldLib.dll", EntryPoint = "HelloWorld")]
13         public static extern void CustomName(string name);
14         static void Main(string[] args)
15         {
16             HelloWorld("LiLi");
17             //跟上面是一样的
18             CustomName("QiQi");
19         }
20     }
21 }

运行程序,结果如下:

这样就成功创建了一个C#可以调用的C++库

下面我们动态调用C++库,这里委托的作用就比较明显了。把委托比喻为C++的函数指针,一点也不为过。

我们在C++库中再新增一个函数GetYear(),用来获取当前年份。

1 int GetYear();
2 
3 int GetYear()
4 {
5     SYSTEMTIME tm;
6     GetLocalTime(&tm);
7 
8     return tm.wYear;
9 }

记得在导出文件中(Source.def)增加GetYear。编译,生成新的DLL

再新建一个C#控制台程序

代码如下:

 1 using System;2 using System.Runtime.InteropServices;3 4 namespace ConsoleApplication35 {6 7     class Program8     {9         [DllImport("kernel32.dll")]
10         public static extern IntPtr LoadLibrary(string lpFileName);
11 
12         [DllImport("kernel32.dll")]
13         public static extern IntPtr GetProcAddress(IntPtr hModule, string lpProcName);
14 
15         [DllImport("kernel32", EntryPoint = "FreeLibrary", SetLastError = true)]
16         public static extern bool FreeLibrary(IntPtr hModule);
17 
18         //声明委托,这里的签名,需要跟C++库中的对应
19         delegate int GetYearDelegate();
20 
21         static void Main(string[] args)
22         {
23             GetYearDelegate m_fGetYear;
24             IntPtr hModule = LoadLibrary("HelloWorldLib.dll");
25             if(hModule != IntPtr.Zero)
26             {
27                 IntPtr hProc = GetProcAddress(hModule, "GetYear");
28                 if(hProc != IntPtr.Zero)
29                 {
30                     m_fGetYear = (GetYearDelegate)Marshal.GetDelegateForFunctionPointer(hProc, typeof(GetYearDelegate));
31 
32                     //在这里可以调用
33                     int year = m_fGetYear();
34                     Console.WriteLine("年份是:" + year);
35                 }
36             }
37         }
38     }
39 }

运行结果:

好的,前面函数里面涉及的都是简单数据类型,下面来介绍一下复杂数据类型。这里指的是结构体

在C++库中定义一个GetDate()的函数,代码如下。这里也要记得在导出文件中添加(Source.def)

struct MyDate
{int year;int month;int day;
};MyDate GetDate();MyDate GetDate()
{SYSTEMTIME tm;GetLocalTime(&tm);MyDate md;md.day = tm.wDay;md.month = tm.wMonth;md.year = tm.wYear;return md;
}

 新建一个C#控制台程序,完整代码如下

 1 using System;2 using System.Runtime.InteropServices;3 4 namespace ConsoleApplication35 {  6     struct MyDate7     {8         public int Year;9         public int Month;
10         public int Day;
11     }
12 
13 
14     class Program
15     {
16         [DllImport("kernel32.dll")]
17         public static extern IntPtr LoadLibrary(string lpFileName);
18 
19         [DllImport("kernel32.dll")]
20         public static extern IntPtr GetProcAddress(IntPtr hModule, string lpProcName);
21 
22         [DllImport("kernel32", EntryPoint = "FreeLibrary", SetLastError = true)]
23         public static extern bool FreeLibrary(IntPtr hModule);
24 
25         delegate IntPtr GetDateDelegate();
26 
27         static void Main(string[] args)
28         {
29             GetDateDelegate m_fGetDate;
30             IntPtr hModule = LoadLibrary("HelloWorldLib.dll");
31 
32             if (hModule != IntPtr.Zero)
33             {
34                 IntPtr hProc = GetProcAddress(hModule, "GetDate");
35                 if (hProc != IntPtr.Zero)
36                 {
37                     m_fGetDate = (GetDateDelegate)Marshal.GetDelegateForFunctionPointer(hProc, typeof(GetDateDelegate));
38                     IntPtr ptr = m_fGetDate();
39                     if(ptr != IntPtr.Zero)
40                     {
41                         MyDate md = (MyDate)Marshal.PtrToStructure(ptr, typeof(MyDate));
42                         Console.WriteLine("{0}年-{1}月-{2}日",md.Year,md.Month,md.Day);
43                     }
44                 }
45             }
46         }
47     }
48 }

运行结果如下:

C#与C++互操作,很重要的一个地方就是,要注意数据类型的对应。有时还需要加上一些限制,

关于C#与C++数据类型对应

可以参考以下链接:

https://www.cnblogs.com/zjoch/p/5999335.html

大部分硬件厂商提供的SDK都是需要C++来调用的,有了上面的知识,使用C#来调用一些硬件的SDK就比较容易了。只需要使用C++再进行一次封装就行了。

二、C++调用C#库

这里用到是C++/CLI,就是如何用C++在·NET中编程。就是因为有这个东西的存在,C++才能调用C#的库

下面新建一个C#类库CSharpLib

这里我们使用C#封装一个读取XML节点的函数(仅供演示)

先创建一个XML文件,并保存为Student.xml

1 <?xml version="1.0" encoding="UTF-8" standalone="yes"?> 
2 <Student>
3     <Name>自由在高处</Name>
4     <Age>17</Age>
5     <Size>40</Size>
6 </Student>

在CSharpLib工程中创建一个XmlQuery类(包含XPathQuery和GetFirstStudent两个成员函数)和一个Student结构体,代码如下:

 1 public class XmlQuery2     {3         private string fileName;4         private XDocument doc;5 6         public XmlQuery(string fileName)7         {8             this.fileName = fileName;9             doc = XDocument.Load(fileName);
10         }
11 
12         public XmlQuery()
13         {
14 
15         }
16 
17         public string XPathQuery(string xPath)
18         {
19             if (doc == null)
20                 return "";
21 
22             var result = doc.XPathSelectElement(xPath);
23 
24             if (result == null)
25                 return "";
26 
27             return result.Value;
28         }
29 
30         public Student GetFirstStudent()
31         {
32             if (doc == null)
33                 return new Student();
34 
35             var root = doc.Root;
36 
37             Student student = new Student();
38             student.Name = root.Element("Name").Value;
39             student.Age = Convert.ToInt32( root.Element("Age").Value);
40             student.Size = Convert.ToInt32(root.Element("Size").Value);
41 
42             return student;
43         }
44     }
45 
46     public struct Student
47     {
48         /// <summary>
49         /// 需要限定长度,否则会转换失败
50         /// </summary>
51         [MarshalAs(UnmanagedType.ByValTStr, SizeConst = 256)]
52         public string Name;
53 
54         /// <summary>
55         /// 基本数据类型注意封送时对应的类型即可
56         /// </summary>
57         public int Age;
58 
59         public int Size;
60     }

然后我们创建一个C++控制台程序UseCSharpLib

创建完成后,将前面编译的CSharpLib.dll和Student.xml拷贝到编译目录下和代码目录下

然后到属性页里开启公共语言运行时支持

使用#using引用前面编译并复制到代码目录下的C#库。这里的相对路径是相对代码工程文件所在的位置

1 #using "CSharpLib.dll"

包含必要的头文件及引入命名空间

1 #include<msclr\marshal_cppstd.h>
2 
3 using namespace System;
4 using namespace msclr::interop;
5 using namespace std;
6 using namespace CSharpLib;

使用gcnew实例化对象,并调用XmlQuery类的成员函数XPathQuery:

1 XmlQuery^ query = gcnew XmlQuery("Student.xml");
2 System::String ^str = query->XPathQuery("Student/Name");

XPathQuery函数返回类型是System.String类型,可以直接使用.Net中String类提供的功能,也可以转换成char*来进行下一步的操作

 1 //直接使用System.String类中的属性和函数2 System::Console::WriteLine(str);3 System::Console::WriteLine(str->Length);4     5 //将System.String转换成char*6 System::IntPtr ptr = System::Runtime::InteropServices::Marshal::StringToHGlobalAnsi(str);7 char* chData = (char *)(void *)ptr;8     9 cout << chData << endl;
10 cout << strlen(chData) << endl;
11 
12 //释放内存
13 System::Runtime::InteropServices::Marshal::FreeHGlobal(ptr);

运行结果如下:

GetFirstStudent函数返回了一个Student结构体,下面的代码演示了调用该函数后,将返回值转换为C++中的结构体。

首先在C++中定义一个结构体:

1 struct StudentCPP
2 {
3     char Name[256]; //大小要跟C#中的保持一致
4     int Age;
5     int Size;
6 };

调用GetFirstStudent

 1 CSharpLib::Student stu = query->GetFirstStudent();2 3 //可以直接操作stu4 int age = stu.Age;5 System::String^ name = stu.Name;6 int size = stu.Size;7 8 //也可以转换为C++中的结构体9 System::IntPtr stuPtr = System::Runtime::InteropServices::Marshal::AllocHGlobal(System::Runtime::InteropServices::Marshal::SizeOf(stu));
10 System::Runtime::InteropServices::Marshal::StructureToPtr(stu, stuPtr,true);
11 StudentCPP* student = (StudentCPP*)(void*)stuPtr;
12 
13 cout << "Name: " << student->Name << endl
14     << "Age: " << student->Age << endl
15     << "Size: " << student->Size << endl;
16 
17 //释放内存
18 System::Runtime::InteropServices::Marshal::FreeHGlobal(stuPtr);

运行如果如下:

除了C++/CLI这种方试,还有两种方式可以实现C++调用C#,但是这里没做详细介绍了,使用前面的方法基本能满足工作需求了。

向 COM 公开 .NET Core 组件

可以参考以下的链接:

向 COM 公开 .NET Core 组件 - .NET | Microsoft Learn

编写自定义 .NET 主机以从本机代码控制 .NET 运行时

可以参考以下的链接:

编写自定义 .NET 运行时主机 - .NET | Microsoft Learn

说明:

1、需要注意C#和C++在进行互操作时的数据类型对应 。可以参考以下链接

https://www.cnblogs.com/zhaotianff/p/12896297.html

2、C#中的类和结构体可以转换成C++中的结构体,但要注意一些原则,可以参考以下链接

https://www.cnblogs.com/zhaotianff/p/12510286.html

https://www.cnblogs.com/zhaotianff/p/13300438.html

3、暂时只想到这两点

最后再附上示例代码,玩得愉快。

版权声明:

本网仅为发布的内容提供存储空间,不对发表、转载的内容提供任何形式的保证。凡本网注明“来源:XXX网络”的作品,均转载自其它媒体,著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处。

我们尊重并感谢每一位作者,均已注明文章来源和作者。如因作品内容、版权或其它问题,请及时与我们联系,联系邮箱:809451989@qq.com,投稿邮箱:809451989@qq.com