C#和C++混合编程

由于历史原因,很多时候我们的代码并不完全是使用.NET写成的。这时候和以往C++代码的混合编程就显得相当重要了。最近碰到了这样的问题,将方法简要记述如下。

调用简单的C++函数

要在C#代码中调用C++函数,大体的思路是这样的:首先将C++函数写成DLL形式的库,然后在C#中导入DLL中的函数进行调用。具体的代码类似这样:
C++代码:

1 int StaticElementNumber = 10;
2  extern "C" AFX_API_EXPORT int GetArrayElementNumber()
3 {
4 return StaticElementNumber;
5 }

C#代码:
(导入函数部分,写在调用函数所在类中)

1 [DllImport("MFCDll.dll")]
2  public static extern int GetArrayElementNumber();
3  int ElementNumber = GetArrayElementNumber();

s其中的细节,比如int和char等数据类型在C++和C#中占用的空间不同等等CLR会自动处理。(主要是通过Marshal类自动处理)

这样的调用还支持调试。打开C#工程的Properties,在Debug选项卡中勾选Enable unmanaged code debugging即可启用C++代码调试。这样在调试模式下,调用这个函数时可以继续按F11跟进函数内部进行调试。

传递GDI对象

一些复杂的Windows对象可以通过句柄来传送。比如下面的代码就将一个GDI+ Bitmap对象转换成GDI句柄进行传送。
C++代码(GDI+的声明,引用等等省略):

复制代码
1 extern "C" AFX_API_EXPORT HBITMAP GetABitmap(WCHAR *strFileName)
2 {
3 Gdiplus::GdiplusStartupInput gdiplusStartupInput;
4 ULONG_PTR gdiplusToken;
5 GdiplusStartup(&gdiplusToken, &gdiplusStartupInput, NULL);
6 Bitmap *bitmap = Bitmap::FromFile(strFileName);
7 HBITMAP HBitmapToReturn;
8 bitmap->GetHBITMAP(NULL, &HBitmapToReturn);
9 GdiplusShutdown(gdiplusToken);
10
11 return HBitmapToReturn;
12 }
复制代码

C#代码(用户界面采用WPF,略去相关声明和引用):

复制代码
1 [DllImport("MFCDll.dll")]
2  public static extern IntPtr GetABitmap([MarshalAs(UnmanagedType.LPWStr)] string strFileName);
3
4  private void MenuItemFileOpenOnClicked(object sender, RoutedEventArgs e)
5 {
6 OpenFileDialog dialog = new OpenFileDialog();
7 dialog.Title = "Load an image...";
8 dialog.Multiselect = false;
9 if (dialog.ShowDialog() == true)
10 {
11 mainGrid.Children.Clear();
12
13 IntPtr hBitmap = GetABitmap(dialog.FileName);
14 Bitmap bitmap = Bitmap.FromHbitmap(hBitmap);
15 System.Windows.Controls.Image image = new Windows.Controls.Image();
16 image.Source = Windows.Interop.Imaging.CreateBitmapSourceFromHBitmap(hBitmap, ro, Int32Rect.Empty,
17 Windows.Media.Imaging.BitmapSizeOptions.FromEmptyOptions());
18 image.Stretch = System.Windows.Media.Stretch.Fill;
19 DeleteObject(hBitmap);
20 mainGrid.Children.Add(image);
21 }
22 }
复制代码

 

 

 

传递数组

传递定长数组很简单,此处不述。下面的代码实现变长数组的传递:
C++代码:

复制代码
1 int StaticElementNumber = 10;
2 extern "C" AFX_API_EXPORT bool GetArray(int ElementNumber, double *BaseAddress)
3 {
4 if (ElementNumber < StaticElementNumber)
5 {
6 return false;
7 }
8
9 for (int i = 0; i < StaticElementNumber; ++i)
10 {
11 BaseAddress[i] = 1 / ((double)i + 1);
12 }
13
14 return true;
15 }
16
17 extern "C" AFX_API_EXPORT int GetArrayElementNumber()
18 {
19 return StaticElementNumber;
20 }
复制代码

 

 

C#代码:  

复制代码
1 [DllImport("MFCDll.dll")]
2 public static extern bool GetArray(int ElementNumber, [MarshalAs(UnmanagedType.LPArray, SizeParamIndex = 0)] double[] BaseAddress);
3
4 private void MenuItemFileGetArrayOnClicked(object sender, RoutedEventArgs e)
5 {
6 //Get array data.
7 int ElementNumber = GetArrayElementNumber();
8 double[] doubleArray = new double[ElementNumber];
9 GetArray(ElementNumber, doubleArray);
10
11 //Show the data.
12 mainGrid.Children.Clear();
13 ListBox listBox = new ListBox();
14 foreach (double number in doubleArray)
15 {
16 listBox.Items.Add(number);
17 }
18 mainGrid.Children.Add(listBox);
19 }
20
复制代码

有了这三个功能,一般来说C++代码复用到C#平台上就是比较简单的事情了。

posted on   grapeot  阅读(26642)  评论(3编辑  收藏  举报

编辑推荐:
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
阅读排行:
· 周边上新:园子的第一款马克杯温暖上架
· Open-Sora 2.0 重磅开源!
· .NET周刊【3月第1期 2025-03-02】
· 分享 3 个 .NET 开源的文件压缩处理库,助力快速实现文件压缩解压功能!
· [AI/GPT/综述] AI Agent的设计模式综述

导航

< 2010年1月 >
27 28 29 30 31 1 2
3 4 5 6 7 8 9
10 11 12 13 14 15 16
17 18 19 20 21 22 23
24 25 26 27 28 29 30
31 1 2 3 4 5 6

统计

点击右上角即可分享
微信分享提示