C# 調用C++/MFC寫的dll


  C#調用C++的非托管類的dll其實很簡單基本就是固定的調用格式.

  dll的編寫,首先是打開VS新建一個C++的控制台程序,下一步后選擇dll以及空文檔即可。然后就是添加一個類添加一個方法。方法排頭固定格式 extern"C"__declspec(dllexport) 后面加方法即可。 例如如下代碼:

C++dll代碼:

extern "C" __declspec(dllexport) char* ShowImages(BYTE img[],int w,int h){;}

  C#調用dll基本也是固定格式,如下樣式,這邊有添加unsafe這是因為有用到指針的緣故[沒用指針不需要unsafe],C#用指針必須添加

unsafe,並且在項目屬性里面的生成允許不安全代碼打鈎才行.

  還有一點是C#調用DLL需要添加命名空間

using System.Runtime.InteropServices;

  代碼樣式如下:

C#調用代碼:

[DllImport("dll名字.dll", CharSet = CharSet.Auto, CallingConvention = CallingConvention.Cdecl)]
unsafe public static extern byte* ShowImages(byte[] src, int w, int h);

  按照如上編寫dll以及調用失敗原因多數可能就是C++的C#的數據類型沒有對應的原因。

  最后還有一點要提的是,具有指針返回值的方法,該指針變量在C++中必須是全局的,方法體里的私有的會出錯。原因是出了方法體就被釋放了,雖然是return出來的,也沒有效果。
  這個問題我之前就遇到過, C++里面return出來一個指針地址,然后C#里面調用,出來的數據亂七八糟,當時一直很郁悶,后來終於發現原因就是指針return出來還是會被釋放掉,所以獲得的數據亂七八糟。解決方法當然是把指針變量設成全局變量。

 

補充:[2014/01/16]

關於MFC編寫DLL和C++寫DLL是一樣的;[以下是MFC規則DLL]

唯一不同點是在方法體最前添加一句話:AFX_MANAGE_STATE(AfxGetStaticModuleState());

如:

extern "C" __declspec(dllexport) void ShowImages()
{
  AFX_MANAGE_STATE(AfxGetStaticModuleState());//MFC比普通的dll增加的內容(如何不涉及控件可以不加這句)

  //......
}

 

關於擴展DLL編寫:

新建擴展DLL工程,添加類后,在類的開頭修改 class AFX_EXT_CLASS CFormatBuilder : public CObject{;}

主要就是添加一個AFX_EXT_CLASS這樣這個類就會作為DLL輸出了。

 

C# 調用方法不變...

但MFC/Win32調用MFC/Win32的DLL有一些變化,方法如下:

1、顯示調用:[MQXS.dll復制到運行目錄下]

 typedef void (*pFunction)(void);// 定義函數變量 
 ::HINSTANCE hinstance = ::LoadLibraryW(_T("MQXS.dll"));
  if (hinstance == NULL)   MessageBox(_T("DLL加載失敗"));
 pFunction function = (pFunction)GetProcAddress(hinstance,"ShowImages");
 function();

 2、隱式調用:[規則DLL][MQXS.dll和MQXS.lib復制到運行目錄下]

#include<stdio.h>
#pragma comment(lib,"MQXS.lib") extern "C" _declspec(dllimport) void ShowImages(); ShowImages();//調用

 3、隱式調用:[擴展DLL][MQXS.dll和MQXS.lib以及調用的類的頭文件FormatBuilder.h復制到運行目錄下]

 #include "FormatBuilder.h"
#pragma comment(lib,"MQXSDLL.lib")
CFormatBuilder builder;//之后就可以使用該類的所有方法,全局變量了

 補充:結構體傳輸定義

C++:

typedef struct Student
{
     char name[20];
     int age;
     double scores[32];
}Student;

extern "C" __declspec(dllexport) int GetStudent(Student stu);

C#:

[StructLayout(LayoutKind.Sequential)]
struct Student
{
      [MarshalAs(UnmanagedType.ByValTStr,SizeConst=20)]
      public string name;
      public int age;
      [MarshalAs(UnmanagedType.ByValArray,SizeConst=32)]
      public double[] scores;
}

數組的傳遞:

C++:

extern "C" __declspec(dllexport) void SetData(int *data)

C#:

public static extern void SetData([MarshalAs(UnmanagedType.LPArray, SizeParamIndex = 1)]  ref int[] data);

 

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM